@datagrok/bio 2.0.0 → 2.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/package-test.js +133 -16
- package/dist/package.js +133 -16
- package/package.json +2 -2
- package/src/__jest__/remote.test.ts +3 -2
- package/src/calculations/fingerprints.ts +76 -0
- package/src/package.ts +12 -4
- package/src/utils/cell-renderer.ts +2 -2
- package/{test-Bio-c5a05867bf79-1b9affa5.html → test-Bio-c5a05867bf79-52614433.html} +2 -2
package/dist/package.js
CHANGED
|
@@ -2754,6 +2754,7 @@ class Measure {
|
|
|
2754
2754
|
"use strict";
|
|
2755
2755
|
__webpack_require__.r(__webpack_exports__);
|
|
2756
2756
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
2757
|
+
/* harmony export */ "addCliffsBooleanCol": () => (/* binding */ addCliffsBooleanCol),
|
|
2757
2758
|
/* harmony export */ "getActivityCliffs": () => (/* binding */ getActivityCliffs),
|
|
2758
2759
|
/* harmony export */ "getSimilaritiesFromDistances": () => (/* binding */ getSimilaritiesFromDistances),
|
|
2759
2760
|
/* harmony export */ "getSimilaritiesMarix": () => (/* binding */ getSimilaritiesMarix)
|
|
@@ -2784,7 +2785,7 @@ let zoom = false;
|
|
|
2784
2785
|
const molColumnNames = ['1_seq', '2_seq'];
|
|
2785
2786
|
const nonNormalizedDistances = ['Levenshtein'];
|
|
2786
2787
|
// Searches for activity cliffs in a chemical dataset by selected cutoff
|
|
2787
|
-
function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, activities, similarity, similarityMetric, methodName, semType, tags, seqSpaceFunc, simMatrixFunc, tooltipFunc, propertyPanelFunc,
|
|
2788
|
+
function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, activities, similarity, similarityMetric, methodName, semType, tags, seqSpaceFunc, simMatrixFunc, tooltipFunc, propertyPanelFunc, seqSpaceOptions) {
|
|
2788
2789
|
return __awaiter(this, void 0, void 0, function* () {
|
|
2789
2790
|
const automaticSimilarityLimit = false;
|
|
2790
2791
|
const MIN_SIMILARITY = 80;
|
|
@@ -2798,7 +2799,7 @@ function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, acti
|
|
|
2798
2799
|
methodName: methodName,
|
|
2799
2800
|
similarityMetric: similarityMetric,
|
|
2800
2801
|
embedAxesNames: axesNames,
|
|
2801
|
-
options:
|
|
2802
|
+
options: seqSpaceOptions
|
|
2802
2803
|
});
|
|
2803
2804
|
for (const col of coordinates) {
|
|
2804
2805
|
const listValues = col.toList();
|
|
@@ -2817,12 +2818,15 @@ function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, acti
|
|
|
2817
2818
|
const saliVals = [];
|
|
2818
2819
|
const n1 = [];
|
|
2819
2820
|
const n2 = [];
|
|
2821
|
+
const cliffsMolIds = new Set();
|
|
2820
2822
|
for (let i = 0; i != dim - 1; ++i) {
|
|
2821
2823
|
for (let j = 0; j != dim - 1 - i; ++j) {
|
|
2822
2824
|
const sim = simArr[i] ? simArr[i].get(j) : 0;
|
|
2823
2825
|
if (sim >= optSimilarityLimit) {
|
|
2824
2826
|
n1.push(i);
|
|
2825
2827
|
n2.push(i + j + 1);
|
|
2828
|
+
cliffsMolIds.add(i);
|
|
2829
|
+
cliffsMolIds.add(i + j + 1);
|
|
2826
2830
|
simVals.push(sim);
|
|
2827
2831
|
const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));
|
|
2828
2832
|
if (sim != 1)
|
|
@@ -2853,7 +2857,9 @@ function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, acti
|
|
|
2853
2857
|
}
|
|
2854
2858
|
const sali = datagrok_api_dg__WEBPACK_IMPORTED_MODULE_1__.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);
|
|
2855
2859
|
df.columns.add(sali);
|
|
2860
|
+
const cliffsColName = addCliffsBooleanCol(df, cliffsMolIds);
|
|
2856
2861
|
const view = datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.shell.getTableView(df.name);
|
|
2862
|
+
view.grid.columns.byName(cliffsColName).visible = false;
|
|
2857
2863
|
const sp = view.addViewer(datagrok_api_dg__WEBPACK_IMPORTED_MODULE_1__.Viewer.scatterPlot(df, {
|
|
2858
2864
|
xColumnName: axesNames[0],
|
|
2859
2865
|
yColumnName: axesNames[1],
|
|
@@ -2901,10 +2907,12 @@ function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, acti
|
|
|
2901
2907
|
const listCliffsLink = datagrok_api_ui__WEBPACK_IMPORTED_MODULE_2__.button(`${linesRes.linesDf.rowCount} cliffs`, () => {
|
|
2902
2908
|
datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);
|
|
2903
2909
|
});
|
|
2904
|
-
listCliffsLink
|
|
2905
|
-
|
|
2906
|
-
|
|
2907
|
-
|
|
2910
|
+
editSpLinkStyle(listCliffsLink, sp, 10);
|
|
2911
|
+
const filterCliffsButton = datagrok_api_ui__WEBPACK_IMPORTED_MODULE_2__.switchInput(`Show only cliffs`, false);
|
|
2912
|
+
filterCliffsButton.onChanged(() => {
|
|
2913
|
+
filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);
|
|
2914
|
+
});
|
|
2915
|
+
editSpLinkStyle(filterCliffsButton.root, sp, 30);
|
|
2908
2916
|
let timer;
|
|
2909
2917
|
canvas.addEventListener('mousemove', function (event) {
|
|
2910
2918
|
clearTimeout(timer);
|
|
@@ -2951,12 +2959,19 @@ function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, acti
|
|
|
2951
2959
|
}, 300);
|
|
2952
2960
|
zoom = false;
|
|
2953
2961
|
}
|
|
2962
|
+
filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);
|
|
2954
2963
|
});
|
|
2955
2964
|
sp.addProperty('similarityLimit', 'double', optSimilarityLimit);
|
|
2956
2965
|
acc = createPopertyPanel();
|
|
2957
2966
|
return sp;
|
|
2958
2967
|
});
|
|
2959
2968
|
}
|
|
2969
|
+
function editSpLinkStyle(el, sp, topMargin) {
|
|
2970
|
+
el.style.position = 'absolute';
|
|
2971
|
+
el.style.top = `${topMargin}px`;
|
|
2972
|
+
el.style.right = '10px';
|
|
2973
|
+
sp.root.append(el);
|
|
2974
|
+
}
|
|
2960
2975
|
function createPopertyPanel() {
|
|
2961
2976
|
const acc = datagrok_api_ui__WEBPACK_IMPORTED_MODULE_2__.accordion();
|
|
2962
2977
|
const accIcon = datagrok_api_ui__WEBPACK_IMPORTED_MODULE_2__.element('i');
|
|
@@ -3030,11 +3045,11 @@ function renderLines(sp, xAxis, yAxis, linesRes, saliVals, saliOpacityCoef, sali
|
|
|
3030
3045
|
return lines;
|
|
3031
3046
|
}
|
|
3032
3047
|
function createLines(n1, n2, seq, activities, saliVals, simVals, semType, tags) {
|
|
3033
|
-
const lines =
|
|
3048
|
+
const lines = new Array(n1.length).fill(null);
|
|
3034
3049
|
for (let i = 0; i < n1.length; i++) {
|
|
3035
3050
|
const num1 = n1[i];
|
|
3036
3051
|
const num2 = n2[i];
|
|
3037
|
-
lines
|
|
3052
|
+
lines[i] = ({ id: i, mols: [num1, num2], selected: false, a: [], b: [] });
|
|
3038
3053
|
}
|
|
3039
3054
|
const linesDf = datagrok_api_dg__WEBPACK_IMPORTED_MODULE_1__.DataFrame.create(lines.length);
|
|
3040
3055
|
molColumnNames.forEach((it, idx) => {
|
|
@@ -3074,7 +3089,14 @@ function getSimilaritiesFromDistances(dim, distances, simArr) {
|
|
|
3074
3089
|
}
|
|
3075
3090
|
return simArr;
|
|
3076
3091
|
}
|
|
3077
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"activity-cliffs.js","sourceRoot":"","sources":["activity-cliffs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { Matrix } from '@datagrok-libraries/utils/src/type-declarations';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport {removeEmptyStringRows} from '@datagrok-libraries/utils/src/dataframe-utils'\n\nexport interface ILine {\n  id: number;\n  mols: number[];\n  selected: boolean;\n  a: number[]; // [x, y]\n  b: number[]; // [x, y]\n}\n\ninterface IRenderedLines {\n  lines: ILine[];\n  linesDf: DG.DataFrame;\n}\n\nexport interface ISequenceSpaceParams {\n  seqCol: DG.Column,\n  methodName: string,\n  similarityMetric: string,\n  embedAxesNames: string[],\n  options?: any\n}\n\nexport interface ISequenceSpaceResult {\n  distance: Matrix;\n  coordinates: DG.ColumnList;\n}\n\nexport interface ITooltipAndPanelParams {\n  cashedData: any,\n  line: ILine,\n  df: DG.DataFrame,\n  seqCol: DG.Column,\n  activityCol: DG.Column,\n  sali?: number\n}\n\nlet zoom = false;\n\nconst molColumnNames = ['1_seq', '2_seq'];\n\nconst nonNormalizedDistances = ['Levenshtein'];\n\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport async function getActivityCliffs(\n    df: DG.DataFrame, \n    seqCol: DG.Column,\n    encodedCol: DG.Column | null,\n    axesNames: string[],\n    scatterTitle: string,\n    activities: DG.Column, \n    similarity: number,\n    similarityMetric: string, \n    methodName: string, \n    semType: string,\n    tags: {[index: string]: string},\n    seqSpaceFunc: (params: ISequenceSpaceParams) => Promise<ISequenceSpaceResult>,\n    simMatrixFunc: (dim: number, seqCol: DG.Column, df: DG.DataFrame, colName: string, simArr: DG.Column[]) => Promise<DG.Column[]>,\n    tooltipFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    propertyPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    options?: any) : Promise<DG.Viewer> {\n  const automaticSimilarityLimit = false;\n  const MIN_SIMILARITY = 80;\n\n  const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n\n  const dimensionalityReduceCol = encodedCol ?? seqCol;\n  const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n    //@ts-ignore\n  const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n\n  const {distance, coordinates} = await seqSpaceFunc({\n    seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name)!,\n    methodName: methodName,\n    similarityMetric: similarityMetric,\n    embedAxesNames: axesNames,\n    options: options\n  });\n\n  for (const col of coordinates) {\n      const listValues = col.toList();\n      emptyValsIdxs.forEach((ind: number) => listValues.splice(ind, 0, null));\n      df.columns.add(DG.Column.fromList('double', col.name, listValues));\n  }\n\n  const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n  const dim = dimensionalityReduceCol.length;\n  const simArr: DG.Column[] = Array(dim - 1);\n\n  if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n    await simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n  else\n    getSimilaritiesFromDistances(dim, distance, simArr);\n\n  const optSimilarityLimit = initialSimilarityLimit;\n\n  const simVals: number[] = [];\n  const saliVals: number[] = [];\n  const n1: number[] = [];\n  const n2: number[] = [];\n\n  for (let i = 0; i != dim - 1; ++i) {\n    for (let j = 0; j != dim - 1 - i; ++j) {\n      const sim: number = simArr[i] ? simArr[i].get(j) : 0;\n\n      if (sim >= optSimilarityLimit) {\n        n1.push(i);\n        n2.push(i + j + 1);\n        simVals.push(sim);\n        const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n        if (sim != 1)\n          saliVals.push(diff / (1 - sim));\n        else\n          saliVals.push(Infinity);\n      }\n    }\n  }\n\n  const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n  const saliMin = Math.min(...saliValsWithoutInfinity);\n  const saliMax = Math.max(...saliValsWithoutInfinity);\n  const saliOpacityCoef = 0.8/(saliMax - saliMin);\n\n\n  const neighboursCount = new Array(dim).fill(0);\n  const similarityCount = new Array(dim).fill(0);\n  const saliCount = new Array(dim).fill(0);\n\n  for (let i = 0; i != n1.length; ++i) {\n    neighboursCount[n1[i]] += 1;\n    neighboursCount[n2[i]] += 1;\n    similarityCount[n1[i]] += simVals[i];\n    similarityCount[n2[i]] += simVals[i];\n    if (saliVals[i] != Infinity) {\n      if (activities.get(n1[i]) > activities.get(n2[i]))\n        saliCount[n1[i]] += saliVals[i];\n      else\n        saliCount[n2[i]] += saliVals[i];\n    }\n  }\n\n  const sali: DG.Column = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n\n  df.columns.add(sali);\n\n  const view = grok.shell.getTableView(df.name);\n  const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n    xColumnName: axesNames[0],\n    yColumnName: axesNames[1],\n    size: sali.name,\n    color: activities.name,\n    showXSelector: false,\n    showYSelector: false,\n    showSizeSelector: false,\n    showColorSelector: false,\n    markerMinSize: 5,\n    markerMaxSize: 25,\n    title: scatterTitle\n  })) as DG.ScatterPlotViewer;\n\n  const canvas = (sp.getInfo() as any)['canvas'];\n  const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n  const cashedLinesData: any = {};\n  let acc: DG.Accordion;\n\n  linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n    const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n    const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n    sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n    sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n    sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n    linesDfGrid.invalidate();\n    if (line)\n      setTimeout(() => {updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc)}, 1000);      \n  });\n\n  linesRes.linesDf.onSelectionChanged.subscribe((_: any) => {\n    if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n      const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n      line.selected = !line.selected;\n      if (!line.selected)\n        df.selection.setAll(false);\n    }\n    linesRes.lines.forEach((l) => {\n      if (l.selected)\n        l.mols.forEach((m) => df.selection.set(m, true));\n    });\n    linesDfGrid.invalidate();\n  });\n\n  const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n\n  linesDfGrid.onCellClick.subscribe(() => {\n    zoom = true;\n  });\n\n  const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n    grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n  });\n  listCliffsLink.style.position = 'absolute';\n  listCliffsLink.style.top = '10px';\n  listCliffsLink.style.right = '10px';\n  sp.root.append(listCliffsLink);\n\n  let timer: NodeJS.Timeout;\n  canvas.addEventListener('mousemove', function (event: MouseEvent) {\n    clearTimeout(timer);\n    timer = global.setTimeout(function () {\n      const line = checkCursorOnLine(event, canvas, linesRes.lines);\n      if (line && df.mouseOverRowIdx === -1) {\n          ui.tooltip.show(tooltipFunc({cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities}), event.clientX, event.clientY);\n      }\n    }, 500);\n  });\n\n  canvas.addEventListener('mousedown', function(event: MouseEvent) {\n    const line = checkCursorOnLine(event, canvas, linesRes.lines);\n    if (line && df.mouseOverRowIdx === -1) {\n      if (event.ctrlKey) {\n        line.selected = !line.selected;\n        linesRes.linesDf.selection.set(line.id, line.selected);\n\n        if (!line.selected)\n          df.selection.setAll(false);\n\n        linesRes.lines.forEach((l) => {\n          if (l.selected)\n            l.mols.forEach((m) => df.selection.set(m, true));\n        });\n      } else {\n        if (linesRes.linesDf.currentRowIdx !== line.id) {\n          linesRes.linesDf.currentRowIdx = line.id;\n          df.currentRowIdx = line.mols[0];\n          df.selection.set(0, !linesRes.lines[0].selected);\n          df.selection.set(0, linesRes.lines[0].selected);\n        }\n      }\n      const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n      linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n    }\n  });\n\n  sp.onEvent('d4-before-draw-scene')\n    .subscribe((_: any) => {\n      const lines = renderLines(sp,\n        axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n      if (zoom) {\n        const currentLine = lines[linesRes.linesDf.currentRowIdx];\n        setTimeout(()=> {\n          const {zoomLeft, zoomRight, zoomTop, zoomBottom} = getZoomCoordinates(\n            sp.viewport.width,\n            sp.viewport.height,\n            sp.dataFrame.get(axesNames[0], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[0], currentLine.mols[1]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[1])\n          )        \n          sp.zoom(zoomLeft,\n            zoomTop,\n            zoomRight,\n            zoomBottom);\n        }, 300);\n        zoom = false;\n      }\n    });\n\n  sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n  acc = createPopertyPanel();\n  return sp;\n}\n\nfunction createPopertyPanel(): DG.Accordion {\n  const acc = ui.accordion();\n  const accIcon = ui.element('i');\n  accIcon.className = 'grok-icon svg-icon svg-view-layout';\n  acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n  acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n  grok.shell.o = acc.root;\n  return acc;\n}\n\nfunction updatePropertyPanel(\n  df: DG.DataFrame,\n  acc: DG.Accordion,\n  cashedData: any,\n  line: ILine, \n  seqCol: DG.Column, \n  activities: DG.Column,\n  sali: number,\n  propPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement){\n  const panel = acc.getPane('Cliff Details');\n  ui.empty(panel.root);\n  const panelElement = propPanelFunc({cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali});\n  panel.root.append(panelElement);\n  setTimeout(() => {\n    grok.shell.o = acc.root;\n  }, 500);\n}\n\nfunction getZoomCoordinates(W0: number, H0: number, x1: number, y1: number, x2: number, y2: number) {\n  const W1 = Math.abs(x1 - x2);\n  const H1 = Math.abs(y1 - y2);\n  const scaleW = W0 / W1;\n  const scaleH = H0 / H1;\n  const scale = Math.min(scaleW, scaleH);\n  const W2 = (W0 / scale) * 5;\n  const H2 = (H0 / scale) * 5;\n  const left = x1 < x2 ? x1 : x2;\n  const top = y1 > y2 ? y1 : y2;\n  const zoomLeft = (left + W1 / 2) - W2 / 2;\n  const zoomRight = zoomLeft + W2;\n  const zoomTop = (top - H1 / 2) + H2 / 2;\n  const zoomBottom = zoomTop - H2;\n  return {zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom};\n}\n\nfunction checkCursorOnLine(event: any, canvas: any, lines: ILine[]): ILine | null {\n  const rect = canvas.getBoundingClientRect();\n  const x = event.clientX - rect.left;\n  const y = event.clientY - rect.top;\n  let closestLine = null;\n  let minDist = 0;\n  for (const line of lines) {\n    const dist =\n      Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n      Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n    if ((!minDist && dist < 2) || dist < minDist) {\n      minDist = dist;\n      closestLine = line;\n    }\n  }\n  return closestLine;\n}\n\nfunction renderLines(sp: DG.ScatterPlotViewer,\n  xAxis: string, yAxis: string, linesRes: IRenderedLines, saliVals: number[], saliOpacityCoef: number, saliMin: number): ILine [] {\n  const lines = linesRes.lines;\n  const canvas = (sp.getInfo() as {[index: string] : any})['canvas'];\n  const ctx = canvas.getContext('2d') as CanvasRenderingContext2D;\n  const x = sp.dataFrame!.columns.byName(xAxis);\n  const y = sp.dataFrame!.columns.byName(yAxis);\n  for (let i = 0; i < lines.length; i++) {\n    const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n    const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n    lines[i].a = [pointFrom.x, pointFrom.y];\n    lines[i].b = [pointTo.x, pointTo.y];\n    const line = new Path2D();\n    line.moveTo(lines[i].a[0], lines[i].a[1]);\n    const color = lines[i].selected ? '255,255,0' : '0,128,0';\n    const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin)*saliOpacityCoef;\n    ctx.strokeStyle = `rgba(${color},${opacity})`;\n    ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n    line.lineTo(lines[i].b[0], lines[i].b[1]);\n    ctx.stroke(line);\n  }\n  return lines;\n}\n\nfunction createLines(\n  n1: number[], \n  n2: number[], \n  seq: DG.Column, \n  activities: DG.Column, \n  saliVals: number[],\n  simVals: number[],\n  semType: string,\n  tags: {[index: string]: string}) : IRenderedLines {\n  const lines: ILine[] = [];\n  for (let i = 0; i < n1.length; i++) {\n    const num1 = n1[i];\n    const num2 = n2[i];\n    lines.push({id: i, mols: [num1, num2], selected: false, a: [], b: []});\n  }\n  const linesDf = DG.DataFrame.create(lines.length);\n  molColumnNames.forEach((it, idx) => {\n    linesDf.columns.addNewString(it).init((i: number) => seq.get(lines[i].mols[idx]));\n    setTags(linesDf.col(it)!, tags);\n    linesDf.col(it)!.semType = semType;\n  })\n  linesDf.columns.addNewFloat('act_diff')\n    .init((i: number) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n  linesDf.columns.addNewInt('line_index').init((i: number) => i);\n  linesDf.columns.addNewFloat('sali').init((i: number) => saliVals[i]);\n  linesDf.columns.addNewFloat('sim').init((i: number) => simVals[i]);\n  return {lines, linesDf};\n}\n\nfunction setTags(col: DG.Column, tags: {[index: string]: string}) {\n  Object.keys(tags).forEach(tag => {\n    col.tags[tag] = tags[tag];\n  })\n}\n\nexport async function getSimilaritiesMarix(\n    dim: number, \n    seqCol: DG.Column, \n    dfSeq: DG.DataFrame, \n    simArr: DG.Column[],\n    simFunc: (col: DG.Column, mol: string) => Promise<DG.Column | null>\n    )\n  : Promise<DG.Column[]> {\n  for (let i = 0; i != dim - 1; ++i) {\n    const mol = seqCol.get(i);\n    dfSeq.rows.removeAt(0, 1, false);\n    simArr[i] = (await simFunc(dfSeq.col('seq')!, mol))!;\n  }\n  return simArr;\n}\n\n\nexport function getSimilaritiesFromDistances(dim: number, distances: Matrix, simArr: DG.Column[])\n  : DG.Column[] {\n  for (let i = 0; i < dim - 1; ++i) {\n    const similarityArr = new Array(dim - i - 1).fill(0);\n    for (let j = i + 1; j < dim; ++j) {\n      similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n    }\n    simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n  }\n  return simArr;\n}\n\n"]}
|
|
3092
|
+
function addCliffsBooleanCol(df, ids) {
|
|
3093
|
+
const colname = 'containsCliff';
|
|
3094
|
+
const colNameInd = df.columns.names().filter((it) => it.includes(colname)).length + 1;
|
|
3095
|
+
const newColName = `${colname}_${colNameInd}`;
|
|
3096
|
+
df.columns.addNewBool(newColName).init((i) => ids.has(i));
|
|
3097
|
+
return newColName;
|
|
3098
|
+
}
|
|
3099
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"activity-cliffs.js","sourceRoot":"","sources":["activity-cliffs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { Matrix } from '@datagrok-libraries/utils/src/type-declarations';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport {removeEmptyStringRows} from '@datagrok-libraries/utils/src/dataframe-utils'\n\nexport interface ILine {\n  id: number;\n  mols: number[];\n  selected: boolean;\n  a: number[]; // [x, y]\n  b: number[]; // [x, y]\n}\n\ninterface IRenderedLines {\n  lines: ILine[];\n  linesDf: DG.DataFrame;\n}\n\nexport interface ISequenceSpaceParams {\n  seqCol: DG.Column,\n  methodName: string,\n  similarityMetric: string,\n  embedAxesNames: string[],\n  options?: any\n}\n\nexport interface ISequenceSpaceResult {\n  distance: Matrix;\n  coordinates: DG.ColumnList;\n}\n\nexport interface ITooltipAndPanelParams {\n  cashedData: any,\n  line: ILine,\n  df: DG.DataFrame,\n  seqCol: DG.Column,\n  activityCol: DG.Column,\n  sali?: number\n}\n\nlet zoom = false;\n\nconst molColumnNames = ['1_seq', '2_seq'];\n\nconst nonNormalizedDistances = ['Levenshtein'];\n\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport async function getActivityCliffs(\n    df: DG.DataFrame, \n    seqCol: DG.Column,\n    encodedCol: DG.Column | null,\n    axesNames: string[],\n    scatterTitle: string,\n    activities: DG.Column, \n    similarity: number,\n    similarityMetric: string, \n    methodName: string, \n    semType: string,\n    tags: {[index: string]: string},\n    seqSpaceFunc: (params: ISequenceSpaceParams) => Promise<ISequenceSpaceResult>,\n    simMatrixFunc: (dim: number, seqCol: DG.Column, df: DG.DataFrame, colName: string, simArr: DG.Column[]) => Promise<DG.Column[]>,\n    tooltipFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    propertyPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    seqSpaceOptions?: any) : Promise<DG.Viewer> {\n\n  const automaticSimilarityLimit = false;\n  const MIN_SIMILARITY = 80;\n\n  const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n\n  const dimensionalityReduceCol = encodedCol ?? seqCol;\n  const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n    //@ts-ignore\n  const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n\n  const {distance, coordinates} = await seqSpaceFunc({\n    seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name)!,\n    methodName: methodName,\n    similarityMetric: similarityMetric,\n    embedAxesNames: axesNames,\n    options: seqSpaceOptions\n  });\n\n  for (const col of coordinates) {\n      const listValues = col.toList();\n      emptyValsIdxs.forEach((ind: number) => listValues.splice(ind, 0, null));\n      df.columns.add(DG.Column.fromList('double', col.name, listValues));\n  }\n\n  const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n  const dim = dimensionalityReduceCol.length;\n  const simArr: DG.Column[] = Array(dim - 1);\n\n  if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n    await simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n  else\n    getSimilaritiesFromDistances(dim, distance, simArr);\n\n  const optSimilarityLimit = initialSimilarityLimit;\n\n  const simVals: number[] = [];\n  const saliVals: number[] = [];\n  const n1: number[] = [];\n  const n2: number[] = [];\n  const cliffsMolIds = new Set<number>();\n\n  for (let i = 0; i != dim - 1; ++i) {\n    for (let j = 0; j != dim - 1 - i; ++j) {\n      const sim: number = simArr[i] ? simArr[i].get(j) : 0;\n\n      if (sim >= optSimilarityLimit) {\n        n1.push(i);\n        n2.push(i + j + 1);\n        cliffsMolIds.add(i);\n        cliffsMolIds.add(i + j + 1);\n        simVals.push(sim);\n        const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n        if (sim != 1)\n          saliVals.push(diff / (1 - sim));\n        else\n          saliVals.push(Infinity);\n      }\n    }\n  }\n\n  const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n  const saliMin = Math.min(...saliValsWithoutInfinity);\n  const saliMax = Math.max(...saliValsWithoutInfinity);\n  const saliOpacityCoef = 0.8/(saliMax - saliMin);\n\n\n  const neighboursCount = new Array(dim).fill(0);\n  const similarityCount = new Array(dim).fill(0);\n  const saliCount = new Array(dim).fill(0);\n\n  for (let i = 0; i != n1.length; ++i) {\n    neighboursCount[n1[i]] += 1;\n    neighboursCount[n2[i]] += 1;\n    similarityCount[n1[i]] += simVals[i];\n    similarityCount[n2[i]] += simVals[i];\n    if (saliVals[i] != Infinity) {\n      if (activities.get(n1[i]) > activities.get(n2[i]))\n        saliCount[n1[i]] += saliVals[i];\n      else\n        saliCount[n2[i]] += saliVals[i];\n    }\n  }\n\n  const sali: DG.Column = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n\n  df.columns.add(sali);\n  const cliffsColName = addCliffsBooleanCol(df, cliffsMolIds);\n\n  const view = grok.shell.getTableView(df.name);\n  view.grid.columns.byName(cliffsColName)!.visible = false;\n  const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n    xColumnName: axesNames[0],\n    yColumnName: axesNames[1],\n    size: sali.name,\n    color: activities.name,\n    showXSelector: false,\n    showYSelector: false,\n    showSizeSelector: false,\n    showColorSelector: false,\n    markerMinSize: 5,\n    markerMaxSize: 25,\n    title: scatterTitle\n  })) as DG.ScatterPlotViewer;\n\n  const canvas = (sp.getInfo() as any)['canvas'];\n  const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n\n  const cashedLinesData: any = {};\n  let acc: DG.Accordion;\n\n  linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n    const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n    const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n    sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n    sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n    sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n    linesDfGrid.invalidate();\n    if (line)\n      setTimeout(() => {updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc)}, 1000);      \n  });\n\n  linesRes.linesDf.onSelectionChanged.subscribe((_: any) => {\n    if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n      const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n      line.selected = !line.selected;\n      if (!line.selected)\n        df.selection.setAll(false);\n    }\n    linesRes.lines.forEach((l) => {\n      if (l.selected)\n        l.mols.forEach((m) => df.selection.set(m, true));\n    });\n    linesDfGrid.invalidate();\n  });\n\n  const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n\n  linesDfGrid.onCellClick.subscribe(() => {\n    zoom = true;\n  });\n\n  const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n    grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n  });\n  editSpLinkStyle(listCliffsLink, sp, 10);\n\n  const filterCliffsButton = ui.switchInput(`Show only cliffs`, false);\n  filterCliffsButton.onChanged(()=> {\n    filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n  });\n  editSpLinkStyle(filterCliffsButton.root, sp, 30);\n\n  let timer: NodeJS.Timeout;\n  canvas.addEventListener('mousemove', function (event: MouseEvent) {\n    clearTimeout(timer);\n    timer = global.setTimeout(function () {\n      const line = checkCursorOnLine(event, canvas, linesRes.lines);\n      if (line && df.mouseOverRowIdx === -1) {\n          ui.tooltip.show(tooltipFunc({cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities}), event.clientX, event.clientY);\n      }\n    }, 500);\n  });\n\n  canvas.addEventListener('mousedown', function(event: MouseEvent) {\n    const line = checkCursorOnLine(event, canvas, linesRes.lines);\n    if (line && df.mouseOverRowIdx === -1) {\n      if (event.ctrlKey) {\n        line.selected = !line.selected;\n        linesRes.linesDf.selection.set(line.id, line.selected);\n\n        if (!line.selected)\n          df.selection.setAll(false);\n\n        linesRes.lines.forEach((l) => {\n          if (l.selected)\n            l.mols.forEach((m) => df.selection.set(m, true));\n        });\n      } else {\n        if (linesRes.linesDf.currentRowIdx !== line.id) {\n          linesRes.linesDf.currentRowIdx = line.id;\n          df.currentRowIdx = line.mols[0];\n          df.selection.set(0, !linesRes.lines[0].selected);\n          df.selection.set(0, linesRes.lines[0].selected);\n        }\n      }\n      const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n      linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n    }\n  });\n\n  sp.onEvent('d4-before-draw-scene')\n    .subscribe((_: any) => {\n      const lines = renderLines(sp,\n        axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n      if (zoom) {\n        const currentLine = lines[linesRes.linesDf.currentRowIdx];\n        setTimeout(()=> {\n          const {zoomLeft, zoomRight, zoomTop, zoomBottom} = getZoomCoordinates(\n            sp.viewport.width,\n            sp.viewport.height,\n            sp.dataFrame.get(axesNames[0], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[0], currentLine.mols[1]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[1])\n          )        \n          sp.zoom(zoomLeft,\n            zoomTop,\n            zoomRight,\n            zoomBottom);\n        }, 300);\n        zoom = false;\n      }\n      filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n    });\n\n  sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n  acc = createPopertyPanel();\n  return sp;\n}\n\nfunction editSpLinkStyle(el: HTMLElement, sp: DG.ScatterPlotViewer, topMargin: number){\n  el.style.position = 'absolute';\n  el.style.top = `${topMargin}px`;\n  el.style.right = '10px';\n  sp.root.append(el);\n}\n\nfunction createPopertyPanel(): DG.Accordion {\n  const acc = ui.accordion();\n  const accIcon = ui.element('i');\n  accIcon.className = 'grok-icon svg-icon svg-view-layout';\n  acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n  acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n  grok.shell.o = acc.root;\n  return acc;\n}\n\nfunction updatePropertyPanel(\n  df: DG.DataFrame,\n  acc: DG.Accordion,\n  cashedData: any,\n  line: ILine, \n  seqCol: DG.Column, \n  activities: DG.Column,\n  sali: number,\n  propPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement){\n  const panel = acc.getPane('Cliff Details');\n  ui.empty(panel.root);\n  const panelElement = propPanelFunc({cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali});\n  panel.root.append(panelElement);\n  setTimeout(() => {\n    grok.shell.o = acc.root;\n  }, 500);\n}\n\nfunction getZoomCoordinates(W0: number, H0: number, x1: number, y1: number, x2: number, y2: number) {\n  const W1 = Math.abs(x1 - x2);\n  const H1 = Math.abs(y1 - y2);\n  const scaleW = W0 / W1;\n  const scaleH = H0 / H1;\n  const scale = Math.min(scaleW, scaleH);\n  const W2 = (W0 / scale) * 5;\n  const H2 = (H0 / scale) * 5;\n  const left = x1 < x2 ? x1 : x2;\n  const top = y1 > y2 ? y1 : y2;\n  const zoomLeft = (left + W1 / 2) - W2 / 2;\n  const zoomRight = zoomLeft + W2;\n  const zoomTop = (top - H1 / 2) + H2 / 2;\n  const zoomBottom = zoomTop - H2;\n  return {zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom};\n}\n\nfunction checkCursorOnLine(event: any, canvas: any, lines: ILine[]): ILine | null {\n  const rect = canvas.getBoundingClientRect();\n  const x = event.clientX - rect.left;\n  const y = event.clientY - rect.top;\n  let closestLine = null;\n  let minDist = 0;\n  for (const line of lines) {\n    const dist =\n      Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n      Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n    if ((!minDist && dist < 2) || dist < minDist) {\n      minDist = dist;\n      closestLine = line;\n    }\n  }\n  return closestLine;\n}\n\nfunction renderLines(sp: DG.ScatterPlotViewer,\n  xAxis: string, yAxis: string, linesRes: IRenderedLines, saliVals: number[], saliOpacityCoef: number, saliMin: number): ILine [] {\n  const lines = linesRes.lines;\n  const canvas = (sp.getInfo() as {[index: string] : any})['canvas'];\n  const ctx = canvas.getContext('2d') as CanvasRenderingContext2D;\n  const x = sp.dataFrame!.columns.byName(xAxis);\n  const y = sp.dataFrame!.columns.byName(yAxis);\n  for (let i = 0; i < lines.length; i++) {\n    const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n    const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n    lines[i].a = [pointFrom.x, pointFrom.y];\n    lines[i].b = [pointTo.x, pointTo.y];\n    const line = new Path2D();\n    line.moveTo(lines[i].a[0], lines[i].a[1]);\n    const color = lines[i].selected ? '255,255,0' : '0,128,0';\n    const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin)*saliOpacityCoef;\n    ctx.strokeStyle = `rgba(${color},${opacity})`;\n    ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n    line.lineTo(lines[i].b[0], lines[i].b[1]);\n    ctx.stroke(line);\n  }\n  return lines;\n}\n\nfunction createLines(\n  n1: number[], \n  n2: number[], \n  seq: DG.Column, \n  activities: DG.Column, \n  saliVals: number[],\n  simVals: number[],\n  semType: string,\n  tags: {[index: string]: string}) : IRenderedLines {\n  const lines: ILine[] = new Array(n1.length).fill(null);\n  for (let i = 0; i < n1.length; i++) {\n    const num1 = n1[i];\n    const num2 = n2[i];\n    lines[i] = ({id: i, mols: [num1, num2], selected: false, a: [], b: []});\n  }\n  const linesDf = DG.DataFrame.create(lines.length);\n  molColumnNames.forEach((it, idx) => {\n    linesDf.columns.addNewString(it).init((i: number) => seq.get(lines[i].mols[idx]));\n    setTags(linesDf.col(it)!, tags);\n    linesDf.col(it)!.semType = semType;\n  })\n  linesDf.columns.addNewFloat('act_diff')\n    .init((i: number) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n  linesDf.columns.addNewInt('line_index').init((i: number) => i);\n  linesDf.columns.addNewFloat('sali').init((i: number) => saliVals[i]);\n  linesDf.columns.addNewFloat('sim').init((i: number) => simVals[i]);\n  return {lines, linesDf};\n}\n\nfunction setTags(col: DG.Column, tags: {[index: string]: string}) {\n  Object.keys(tags).forEach(tag => {\n    col.tags[tag] = tags[tag];\n  })\n}\n\nexport async function getSimilaritiesMarix(\n    dim: number, \n    seqCol: DG.Column, \n    dfSeq: DG.DataFrame, \n    simArr: DG.Column[],\n    simFunc: (col: DG.Column, mol: string) => Promise<DG.Column | null>\n    )\n  : Promise<DG.Column[]> {\n  for (let i = 0; i != dim - 1; ++i) {\n    const mol = seqCol.get(i);\n    dfSeq.rows.removeAt(0, 1, false);\n    simArr[i] = (await simFunc(dfSeq.col('seq')!, mol))!;\n  }\n  return simArr;\n}\n\n\nexport function getSimilaritiesFromDistances(dim: number, distances: Matrix, simArr: DG.Column[])\n  : DG.Column[] {\n  for (let i = 0; i < dim - 1; ++i) {\n    const similarityArr = new Array(dim - i - 1).fill(0);\n    for (let j = i + 1; j < dim; ++j) {\n      similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n    }\n    simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n  }\n  return simArr;\n}\n\nexport function addCliffsBooleanCol(df: DG.DataFrame, ids: Set<number>): string {\n  const colname = 'containsCliff';\n  const colNameInd = df.columns.names().filter((it: string) => it.includes(colname)).length + 1;\n  const newColName = `${colname}_${colNameInd}`;\n  df.columns.addNewBool(newColName).init((i) => ids.has(i));\n  return newColName;\n}\n\n"]}
|
|
3078
3100
|
|
|
3079
3101
|
/***/ }),
|
|
3080
3102
|
|
|
@@ -4437,6 +4459,92 @@ function jaroWinkler(str1, str2, options) {
|
|
|
4437
4459
|
exports.jaroWinkler = jaroWinkler;
|
|
4438
4460
|
|
|
4439
4461
|
|
|
4462
|
+
/***/ }),
|
|
4463
|
+
|
|
4464
|
+
/***/ "./src/calculations/fingerprints.ts":
|
|
4465
|
+
/*!******************************************!*\
|
|
4466
|
+
!*** ./src/calculations/fingerprints.ts ***!
|
|
4467
|
+
\******************************************/
|
|
4468
|
+
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
4469
|
+
|
|
4470
|
+
"use strict";
|
|
4471
|
+
__webpack_require__.r(__webpack_exports__);
|
|
4472
|
+
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
4473
|
+
/* harmony export */ "getFingerprints": () => (/* binding */ getFingerprints)
|
|
4474
|
+
/* harmony export */ });
|
|
4475
|
+
/* harmony import */ var datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! datagrok-api/grok */ "datagrok-api/grok");
|
|
4476
|
+
/* harmony import */ var datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__);
|
|
4477
|
+
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
4478
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4479
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
4480
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
4481
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
4482
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
4483
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
4484
|
+
});
|
|
4485
|
+
};
|
|
4486
|
+
|
|
4487
|
+
const V2000_ATOM_NAME_POS = 31;
|
|
4488
|
+
function getFingerprints(mols, monomers) {
|
|
4489
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
4490
|
+
const mod = yield datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.functions.call('Chem:getRdKitModule');
|
|
4491
|
+
const fps = [];
|
|
4492
|
+
let dict = new Map();
|
|
4493
|
+
for (let i = 0; i < monomers.length; i++)
|
|
4494
|
+
dict.set(monomers[i], `R${Math.pow(10, (i + 1))}`);
|
|
4495
|
+
mols = changeToV3000(mols, dict);
|
|
4496
|
+
for (let i = 0; i < mols.length; i++) {
|
|
4497
|
+
const mol = mod.get_mol(mols[i]);
|
|
4498
|
+
const fp = mol.get_pattern_fp_as_uint8array();
|
|
4499
|
+
fps.push(fp);
|
|
4500
|
+
mol === null || mol === void 0 ? void 0 : mol.delete();
|
|
4501
|
+
}
|
|
4502
|
+
return fps;
|
|
4503
|
+
});
|
|
4504
|
+
}
|
|
4505
|
+
function changeToV3000(mols, dict) {
|
|
4506
|
+
for (let i = 0; i < mols.length; i++) {
|
|
4507
|
+
let curPos = 0;
|
|
4508
|
+
let endPos = 0;
|
|
4509
|
+
let molV3000 = `
|
|
4510
|
+
Datagrok macromolecule handler
|
|
4511
|
+
|
|
4512
|
+
0 0 0 0 0 0 999 V3000
|
|
4513
|
+
M V30 BEGIN CTAB
|
|
4514
|
+
`;
|
|
4515
|
+
const mol = mols[i];
|
|
4516
|
+
curPos = mol.indexOf('\n', curPos) + 1;
|
|
4517
|
+
curPos = mol.indexOf('\n', curPos) + 1;
|
|
4518
|
+
curPos = mol.indexOf('\n', curPos) + 1;
|
|
4519
|
+
const atomMonomerCounts = parseInt(mol.substring(curPos, curPos + 3));
|
|
4520
|
+
const bondMonomerCounts = parseInt(mol.substring(curPos + 3, curPos + 6));
|
|
4521
|
+
molV3000 += `M V30 COUNTS ${atomMonomerCounts} ${bondMonomerCounts} 0 0 0\n`;
|
|
4522
|
+
molV3000 += 'M V30 BEGIN ATOM\n';
|
|
4523
|
+
for (let atomRowI = 0; atomRowI < atomMonomerCounts; atomRowI++) {
|
|
4524
|
+
curPos = mol.indexOf('\n', curPos) + 1 + V2000_ATOM_NAME_POS;
|
|
4525
|
+
endPos = mol.indexOf(' ', curPos);
|
|
4526
|
+
const monomerName = mol.substring(curPos, endPos);
|
|
4527
|
+
molV3000 += `M V30 ${atomRowI + 1} ${dict.get(monomerName)} 0.000 0.000 0 0\n`;
|
|
4528
|
+
}
|
|
4529
|
+
molV3000 += 'M V30 END ATOM\n';
|
|
4530
|
+
molV3000 += 'M V30 BEGIN BOND\n';
|
|
4531
|
+
for (let bondRowI = 0; bondRowI < bondMonomerCounts; bondRowI++) {
|
|
4532
|
+
curPos = mol.indexOf('\n', curPos) + 1;
|
|
4533
|
+
const firstMonomer = parseInt(mol.substring(curPos, curPos + 3).trim());
|
|
4534
|
+
const secondMonomer = parseInt(mol.substring(curPos + 3, curPos + 6).trim());
|
|
4535
|
+
const order = parseInt(mol.substring(curPos + 6, curPos + 9).trim());
|
|
4536
|
+
molV3000 += `M V30 ${bondRowI + 1} ${order} ${firstMonomer} ${secondMonomer}\n`;
|
|
4537
|
+
}
|
|
4538
|
+
molV3000 += 'M V30 END BOND\n';
|
|
4539
|
+
molV3000 += 'M V30 END CTAB\n';
|
|
4540
|
+
molV3000 += 'M END';
|
|
4541
|
+
console.log(molV3000);
|
|
4542
|
+
mols[i] = molV3000;
|
|
4543
|
+
}
|
|
4544
|
+
return mols;
|
|
4545
|
+
}
|
|
4546
|
+
|
|
4547
|
+
|
|
4440
4548
|
/***/ }),
|
|
4441
4549
|
|
|
4442
4550
|
/***/ "./src/const.ts":
|
|
@@ -5562,10 +5670,10 @@ class MonomerCellRenderer extends datagrok_api_dg__WEBPACK_IMPORTED_MODULE_1__.G
|
|
|
5562
5670
|
g.textBaseline = 'middle';
|
|
5563
5671
|
g.textAlign = 'center';
|
|
5564
5672
|
const palette = getPaletteByType(gridCell.cell.column.getTag(_constants__WEBPACK_IMPORTED_MODULE_0__.TAGS.ALPHABET));
|
|
5565
|
-
const s = gridCell.cell.value
|
|
5673
|
+
const s = gridCell.cell.value || '-';
|
|
5566
5674
|
const color = palette.get(s);
|
|
5567
5675
|
g.fillStyle = color;
|
|
5568
|
-
g.fillText(s, x + (w / 2), y - (h / 2));
|
|
5676
|
+
g.fillText(s, x + (w / 2), y - (h / 2), w);
|
|
5569
5677
|
}
|
|
5570
5678
|
}
|
|
5571
5679
|
class MacromoleculeDifferenceCellRenderer extends datagrok_api_dg__WEBPACK_IMPORTED_MODULE_1__.GridCellRenderer {
|
|
@@ -6864,6 +6972,7 @@ __webpack_require__.r(__webpack_exports__);
|
|
|
6864
6972
|
/* harmony export */ "importFasta": () => (/* binding */ importFasta),
|
|
6865
6973
|
/* harmony export */ "initBio": () => (/* binding */ initBio),
|
|
6866
6974
|
/* harmony export */ "macromoleculeDifferenceCellRenderer": () => (/* binding */ macromoleculeDifferenceCellRenderer),
|
|
6975
|
+
/* harmony export */ "macromoleculesFingerprints": () => (/* binding */ macromoleculesFingerprints),
|
|
6867
6976
|
/* harmony export */ "monomerCellRenderer": () => (/* binding */ monomerCellRenderer),
|
|
6868
6977
|
/* harmony export */ "multipleSequenceAlignmentAny": () => (/* binding */ multipleSequenceAlignmentAny),
|
|
6869
6978
|
/* harmony export */ "panelMSA": () => (/* binding */ panelMSA),
|
|
@@ -6901,6 +7010,7 @@ __webpack_require__.r(__webpack_exports__);
|
|
|
6901
7010
|
/* harmony import */ var _datagrok_libraries_utils_src_dataframe_utils__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! @datagrok-libraries/utils/src/dataframe-utils */ "./node_modules/@datagrok-libraries/utils/src/dataframe-utils.js");
|
|
6902
7011
|
/* harmony import */ var _datagrok_libraries_bio_src_utils_splitter__WEBPACK_IMPORTED_MODULE_18__ = __webpack_require__(/*! @datagrok-libraries/bio/src/utils/splitter */ "./node_modules/@datagrok-libraries/bio/src/utils/splitter.js");
|
|
6903
7012
|
/* harmony import */ var _utils_constants__WEBPACK_IMPORTED_MODULE_19__ = __webpack_require__(/*! ./utils/constants */ "./src/utils/constants.ts");
|
|
7013
|
+
/* harmony import */ var _calculations_fingerprints__WEBPACK_IMPORTED_MODULE_20__ = __webpack_require__(/*! ./calculations/fingerprints */ "./src/calculations/fingerprints.ts");
|
|
6904
7014
|
var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
6905
7015
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
6906
7016
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
@@ -6933,6 +7043,7 @@ const _package = new datagrok_api_dg__WEBPACK_IMPORTED_MODULE_2__.Package();
|
|
|
6933
7043
|
|
|
6934
7044
|
|
|
6935
7045
|
|
|
7046
|
+
|
|
6936
7047
|
//tags: init
|
|
6937
7048
|
function initBio() {
|
|
6938
7049
|
return __awaiter(this, void 0, void 0, function* () {
|
|
@@ -7296,7 +7407,6 @@ function testDetectMacromolecule(path) {
|
|
|
7296
7407
|
//tags: panel, bio
|
|
7297
7408
|
//input: column col {semType: Macromolecule}
|
|
7298
7409
|
function splitToMonomers(col) {
|
|
7299
|
-
var _a;
|
|
7300
7410
|
if (!col.getTag(_datagrok_libraries_bio_src_utils_units_handler__WEBPACK_IMPORTED_MODULE_15__.UnitsHandler.TAGS.aligned).includes(_utils_constants__WEBPACK_IMPORTED_MODULE_19__.MSA))
|
|
7301
7411
|
return datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.shell.error('Splitting is applicable only for aligned sequences');
|
|
7302
7412
|
const tempDf = (0,_datagrok_libraries_bio_src_utils_splitter__WEBPACK_IMPORTED_MODULE_18__.splitAlignedSequences)(col);
|
|
@@ -7307,19 +7417,26 @@ function splitToMonomers(col) {
|
|
|
7307
7417
|
newCol.setTag(datagrok_api_dg__WEBPACK_IMPORTED_MODULE_2__.TAGS.CELL_RENDERER, _utils_constants__WEBPACK_IMPORTED_MODULE_19__.SEM_TYPES.MONOMER);
|
|
7308
7418
|
newCol.setTag(_utils_constants__WEBPACK_IMPORTED_MODULE_19__.TAGS.ALPHABET, col.getTag(_utils_constants__WEBPACK_IMPORTED_MODULE_19__.TAGS.ALPHABET));
|
|
7309
7419
|
}
|
|
7310
|
-
|
|
7420
|
+
datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.shell.tv.grid.invalidate();
|
|
7311
7421
|
}
|
|
7312
7422
|
//name: Bio: getHelmMonomers
|
|
7313
7423
|
//input: column col {semType: Macromolecule}
|
|
7314
|
-
//output: string[] result
|
|
7315
7424
|
function getHelmMonomers(seqCol) {
|
|
7316
|
-
const stats = _datagrok_libraries_bio_src_viewers_web_logo__WEBPACK_IMPORTED_MODULE_4__.WebLogo.getStats(seqCol,
|
|
7425
|
+
const stats = _datagrok_libraries_bio_src_viewers_web_logo__WEBPACK_IMPORTED_MODULE_4__.WebLogo.getStats(seqCol, 1, _datagrok_libraries_bio_src_viewers_web_logo__WEBPACK_IMPORTED_MODULE_4__.WebLogo.splitterAsHelm);
|
|
7317
7426
|
return Object.keys(stats.freq);
|
|
7318
7427
|
}
|
|
7428
|
+
function macromoleculesFingerprints(mcol) {
|
|
7429
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
7430
|
+
datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.functions.call('Chem:getRdKitModule');
|
|
7431
|
+
const monomers = getHelmMonomers(mcol);
|
|
7432
|
+
const mols = yield datagrok_api_grok__WEBPACK_IMPORTED_MODULE_0__.functions.call('HELM:getMolFiles', { mcol: mcol });
|
|
7433
|
+
return (0,_calculations_fingerprints__WEBPACK_IMPORTED_MODULE_20__.getFingerprints)(mols.toList(), monomers);
|
|
7434
|
+
});
|
|
7435
|
+
}
|
|
7319
7436
|
|
|
7320
7437
|
})();
|
|
7321
7438
|
|
|
7322
7439
|
bio = __webpack_exports__;
|
|
7323
7440
|
/******/ })()
|
|
7324
7441
|
;
|
|
7325
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"package.js","mappings":";;;;;;;;;;AAAA;AACA,EAAE,KAA4D;AAC9D,EAAE,CACmD;AACrD,CAAC,sBAAsB;;AAEvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,eAAe;AACjC;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,OAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,iBAAiB,kBAAkB,UAAU;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,eAAe;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;AACA,uBAAuB;AACvB,WAAW;AACX;AACA;AACA,2DAA2D,gBAAgB,IAAI;AAC/E;AACA;AACA;AACA;AACA;AACA,WAAW;AACX,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8DAA8D;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB;AACvB;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,qBAAqB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA,yCAAyC,IAAI;AAC7C,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,aAAa,SAAS,YAAY;AAClD;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+EAA+E;AAC/E;;AAEA,+FAA+F,6BAA6B;AAC5H;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,iCAAiC;AACjC;AACA;AACA,+BAA+B;AAC/B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA,yDAAyD,WAAW;AACpE,6DAA6D,WAAW;AACxE;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,8CAA8C,gCAAgC;;AAE9E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,uEAAuE;AACvE;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA,CAAC;AACD;;;;;;;;;;;;;;;;;;ACrdgD;AACzC,iCAAiC,yDAAc;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,OAAO;AAChE;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qDAAqD,yBAAyB;AAC9E,qDAAqD,yBAAyB;AAC9E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;ACjOK;AACzC,kCAAkC,yDAAc;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;AC3BpC;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC3C8B;AACzE;AACA;AACA;AACA;AACA;AACA,QAAQ,uFAAM,+CAA+C,KAAK;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC;AACzC,YAAY,uFAAM,0CAA0C,YAAY,4BAA4B,IAAI;AACxG;AACA;AACA;AACA,6CAA6C;AAC7C,YAAY,uFAAM,gDAAgD,YAAY,6BAA6B,IAAI;AAC/G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,UAAU;AAC5B;AACA;AACA;AACA;AACA;AACA,wBAAwB,YAAY;AACpC;AACA,YAAY,uFAAM,uCAAuC,KAAK,uBAAuB,SAAS;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,UAAU;AAC3B,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA,YAAY,uFAAM,uCAAuC,KAAK,qBAAqB,MAAM;AACzF;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;AC1JpC;AACP;AACA;AACA;AACA;AACA,wBAAwB,cAAc;AACtC;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACgD;AACzC;AACP;AACO;AACP;AACA;AACA;AACA;AACO;AACP;AACA,+CAA+C,uEAA4B;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,iCAAiC,yDAAc;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC7C3C;AACA;AACO;AACP;AACA;AACA;AACA,CAAC,8BAA8B;AAC/B;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,0BAA0B;AACrC,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,SAAS;AACpB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,SAAS;AACpB;AACA;AACA;AACA;AACA,YAAY,QAAQ;AACpB;AACO,iMAAiM,+BAA+B;AACvO;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACzEL;AACS;AAC/C;AACO;AACP;AACA,sCAAsC;AACtC,mCAAmC;AACnC;AACA;AACA;AACA;AACA,8BAA8B;AAC9B,2BAA2B;AAC3B;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA,qCAAqC,+DAAqB;AAC1D,4BAA4B,+DAAqB;AACjD,8BAA8B,kEAAwB;AACtD;AACA,QAAQ,8EAAkC;AAC1C,gBAAgB,kEAAwB;AACxC;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;AC1DG;AACC;AAC/C;AACO,gCAAgC,wDAAY;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,2EAA4B;AACzD;AACA;AACA;AACA,8BAA8B;AAC9B,kCAAkC;AAClC,6BAA6B;AAC7B;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,+BAA+B;AAC3D;AACA,4CAA4C,yFAA6C;AACzF;AACA;AACA,SAAS;AACT,yBAAyB,uEAA2B;AACpD;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,UAAU;AAC1B;AACA;AACA,8CAA8C;AAC9C,mCAAmC;AACnC,+CAA+C;AAC/C,+BAA+B;AAC/B;AACA;AACA,0BAA0B;AAC1B;AACA,wCAAwC;AACxC,yEAAyE;AACzE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,oFAAwC;AAC/D;AACA,0BAA0B,YAAY,EAAE,GAAG,EAAE,aAAa;AAC1D,SAAS;AACT,kBAAkB,OAAO,EAAE,2BAA2B,EAAE,QAAQ;AAChE;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,eAAe;AAC9B;AACA,gBAAgB,SAAS;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,eAAe;AAC9B,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,eAAe;AAC9B,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA,6BAA6B,qFAAyC;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,gCAAgC;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,eAAe;AAC9B;AACA,gBAAgB,WAAW;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,qFAAyC;AACzD,gBAAgB,yFAA6C;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,2BAA2B;AACvD;AACA;AACA;AACA,6BAA6B,oFAAwC;AACrE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,UAAU;AACzB,eAAe,eAAe;AAC9B,gBAAgB,0BAA0B;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACxNL;AACQ;AACvC;AACP,qBAAqB,2EAA4B;AACjD;AACA,qBAAqB,6DAAmB;AACxC;AACA;AACA;AACA,oBAAoB,wBAAwB;AAC5C;AACA;AACA;AACA;AACA,2BAA2B,mBAAmB;AAC9C;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACpBL;AACQ;AAC9C;AACO;AACP;AACA;AACA,wCAAwC,uDAAa;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,+BAA+B;AAC3D;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,+BAA+B;AAC3D;AACA;AACA;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,sCAAsC;AAClE;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,sCAAsC;AAClE;AACA;AACA;AACA,4BAA4B,kEAAwB;AACpD;AACA,sBAAsB,+DAAgB,SAAS,sEAAuB;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oEAAoE,4EAA6B;AACjG;AACA;AACA,mBAAmB,uDAAa;AAChC;AACA;AACA;AACA,kBAAkB;AAClB,mBAAmB;AACnB,qBAAqB;AACrB,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,cAAc;AAC1E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,oBAAoB;AACpB,eAAe;AACf,cAAc;AACd,cAAc;AACd,kBAAkB;AAClB;AACA;AACA,gBAAgB,cAAc;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,iBAAiB,6BAA6B,WAAW;AAChG;AACA;AACA;AACA;AACA;AACA,eAAe,UAAU;AACzB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,4DAAkB;AAC5C,4BAA4B,kEAAwB;AACpD,yBAAyB,uDAAa;AACtC,yBAAyB,+DAAqB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,WAAW;AAC1B;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,WAAW;AAC1B,eAAe,WAAW;AAC1B,eAAe,WAAW;AAC1B,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,4DAAkB;AAC5C,4BAA4B,kEAAwB;AACpD,yBAAyB,uDAAa;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC3N3C;AACO;AACP;AACA;AACA;AACA;AACA,CAAC,oCAAoC;AACrC;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,cAAc;AAC7B,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AC7B3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AAClB;AACS;AAC4C;AACwB;AAClC;AACG;AAClB;AACM;AACtD;AACA;AACA;AACA;AACA,CAAC,wCAAwC;AACzC,WAAW,YAAY;AACvB,WAAW,QAAQ;AACnB,YAAY,UAAU;AACtB;AACA;AACA;AACA,eAAe,kDAAQ;AACvB;AACA,oEAA0B;AAC1B;AACA;AACO;AACP,wCAAwC,iDAAO;AAC/C;AACA;AACA;AACA;AACO;AACP,iBAAiB;AACjB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,UAAU;AACzB;AACA,+BAA+B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACO,sBAAsB,qDAAW;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,iDAAiD;AAC5D;AACA,WAAW,mDAAmD;AAC9D;AACA,WAAW,mDAAmD;AAC9D;AACA;AACA;AACA;AACA;AACA,8EAA8E,sCAAsC;AACpH,kFAAkF,sCAAsC;AACxH;AACA;AACA;AACA,gFAAgF,oCAAoC;AACpH;AACA;AACA;AACA;AACA,0FAA0F,iBAAiB;AAC3G,mFAAmF,wDAAwD;AAC3I;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,gDAAM;AACjC;AACA,0BAA0B,mDAAS;AACnC;AACA,wBAAwB,gDAAM;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,+BAA+B,2CAAc;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,+CAAQ;AAC7C;AACA;AACA;AACA,qBAAqB;AACrB,oBAAoB,yDAAe,CAAC,gDAAM,EAAE,gDAAM,IAAI,QAAQ,IAAI,gDAAM,IAAI,UAAU;AACtF;AACA;AACA,oBAAoB,yDAAe;AACnC;AACA,aAAa;AACb,+BAA+B,2CAAc;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA,aAAa;AACb,+BAA+B,0DAAgB;AAC/C;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,2BAA2B,sBAAsB,gBAAgB,uBAAuB;AACxF,6BAA6B,kBAAkB,kBAAkB,oBAAoB;AACrF;AACA,iBAAiB,cAAc,KAAK,UAAU,UAAU,0BAA0B,WAAW,aAAa;AAC1G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iDAAiD,uDAAa;AAC9D,qDAAqD,8EAA2B;AAChF;AACA,wCAAwC,+DAAY;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0DAA0D,SAAS;AACnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,eAAe,MAAM;AAC5E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B,SAAS;AACT;AACA;AACA,0CAA0C,cAAc,iCAAiC,cAAc;AACvG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA,0CAA0C,cAAc;AACxD,SAAS;AACT;AACA;AACA;AACA,sBAAsB;AACtB,SAAS;AACT;AACA,oCAAoC,iCAAiC;AACrE,0CAA0C,cAAc,iCAAiC,cAAc;AACvG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,cAAc;AACxD,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,yBAAyB;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,QAAQ,6IAA6I;AAC5M;AACA;AACA;AACA;AACA;AACA,gCAAgC,+DAAY;AAC5C;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,oBAAoB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,cAAc,2BAA2B,UAAU;AACvF;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA,gBAAgB,0DAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,oCAAoC,iDAAO;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8EAA8E,oBAAoB;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,4BAA4B;AAChD;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6C,EAAE,gCAAgC,2BAA2B;AAC1G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,MAAM;AAC3C;AACA,iDAAiD,OAAO;AACxD;AACA;AACA,mDAAmD,aAAa;AAChE;AACA;AACA;AACA,+DAA+D,MAAM;AACrE,wCAAwC,OAAO;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,cAAc;AACvE;AACA,yDAAyD,uBAAuB;AAChF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE;AACA,2CAA2C,mBAAmB;AAC9D,2CAA2C,yBAAyB;AACpE;AACA;AACA,eAAe,WAAW;AAC1B,eAAe,SAAS;AACxB,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA;AACA,iBAAiB,4EAAmC;AACpD,sBAAsB,sEAA6B;AACnD;AACA,iBAAiB,8EAAoC;AACrD,sBAAsB,0EAAgC;AACtD;AACA;AACA;AACA;AACA,iCAAiC,2DAAiB,IAAI,0EAAgC;AACtF,iCAAiC,yDAAgB,IAAI,sEAA6B;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,8DAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA,mEAAmE,kBAAkB;AACrF,eAAe,WAAW;AAC1B,eAAe,QAAQ;AACvB,eAAe,cAAc;AAC7B,iBAAiB,aAAa,wBAAwB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,mFAAM;AAChC,8BAA8B,mFAAM;AACpC,eAAe,iGAAgB,sBAAsB,6FAAY,UAAU,6FAAY;AACvF;AACA;AACA;AACA,kBAAkB,cAAc;AAChC,mBAAmB,WAAW;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,kEAAwB;AAC/E;AACA,qCAAqC,uDAAa;AAClD;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,KAAK;AACpB,gBAAgB,UAAU;AAC1B;AACA;AACA,oBAAoB,yCAAE;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mDAAmD,EAAE;AACrD;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD,OAAO;AACvD;AACA;AACA;AACA,4BAA4B,kEAAwB;AACpD,yDAAyD,kEAAwB,CAAC;AAClF,iCAAiC,uDAAa;AAC9C,qCAAqC,8EAA2B;AAChE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,yCAAE;AAC5B;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,qDAAqD,MAAM,QAAQ,sBAAsB;AACzF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC,IAAI,IAAI;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;AC7xB3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC4E;AACyB;AAC9F;AACP;AACA,8CAA8C,kHAAiC,GAAG,yCAAyC;AAC3H,wFAAwF,0FAAS;AACjG;AACA,KAAK;AACL;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;AClBD;AACY;AAC6B;AACU;AACtF;AACP;AACA,6BAA6B,uGAA0B;AACvD,KAAK;AACL;AACA,uBAAuB,yDAAW;AAClC,wBAAwB,gEAAW;AACnC,KAAK;AACL;AACA,oBAAoB,sGAA2B;AAC/C,gBAAgB,kGAAuB;AACvC,sBAAsB,wGAA6B;AACnD,0BAA0B,+GAAiC;AAC3D,kBAAkB,oGAAyB;AAC3C,sBAAsB,wGAA6B;AACnD,yBAAyB,8GAAgC;AACzD,0BAA0B,+GAAiC;AAC3D,kBAAkB,oGAAyB;AAC3C,iBAAiB,mGAAwB;AACzC,KAAK;AACL;AACO;AACP;AACA;AACA;AACA;AACA;AACA,CAAC,IAAI;AACE;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,oBAAoB;AACnC,gBAAgB,UAAU;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;AC7E3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AACuD;AACP;AACtF;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA,mCAAmC,kEAAwB;AAC3D;AACA,8BAA8B,oGAAqB;AACnD,gBAAgB,wBAAwB;AACxC;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,2BAA2B,4DAAkB;AAC7C;AACA,sBAAsB,kEAAwB,EAAE,4DAAkB;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,cAAc;AACtC,4BAA4B,kBAAkB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,4DAAkB,mBAAmB,sDAAsD;AAChH;AACA,qBAAqB,iEAAuB;AAC5C,kCAAkC,+DAAqB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,oIAAoI;AACvK,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT,+BAA+B,mDAAS,IAAI,2BAA2B;AACvE,YAAY,qEAA2B;AACvC,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,qBAAM;AAC1B;AACA;AACA,oBAAoB,yDAAe,eAAe,0FAA0F;AAC5I;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,2CAA2C;AACvE;AACA,iBAAiB;AACjB;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,gBAAgB,sDAAY;AAC5B,oBAAoB,oDAAU;AAC9B;AACA,iBAAiB,iDAAO,WAAW,kDAAQ;AAC3C,uCAAuC,oDAAU;AACjD,IAAI,sDAAY;AAChB;AACA;AACA;AACA;AACA,IAAI,kDAAQ;AACZ,yCAAyC,iGAAiG;AAC1I;AACA;AACA,QAAQ,sDAAY;AACpB,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kBAAkB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,MAAM,GAAG,QAAQ;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA,qBAAqB,0DAA0D;AAC/E;AACA,oBAAoB,6DAAmB;AACvC;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,KAAK;AACL;AACO;AACP;AACA,wBAAwB,cAAc;AACtC;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACO;AACP,oBAAoB,aAAa;AACjC;AACA,4BAA4B,SAAS;AACrC,uCAAuC,2GAAyB;AAChE;AACA,oBAAoB,oEAA0B;AAC9C;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;ACpTpC;AACP;AACA,0CAA0C,4KAA2C;AACrF;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,8BAA8B,QAAQ,uBAAuB;AAC7D,sBAAsB,0CAA0C;AAChE;AACA,KAAK;AACL;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;ACdL;AACA;AACtC;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA,oBAAoB,oBAAoB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,uBAAuB,6CAAW;AAClC;AACA;AACA;AACA;AACA;AACA,wCAAwC,wDAAc;AACtD;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC5C3C;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACpBN;AAC9B;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACA;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kDAAS;AAC7B;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACpL3C;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACpCE;AACK;AAClD;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,QAAQ;AACnB,YAAY;AACZ;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,aAAa;AACzB;AACA;AACA;AACA,2DAA2D,sDAAM;AACjE;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACO;AACP;AACA,4BAA4B,sDAAM;AAClC;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,sBAAsB,sDAAM;AAC5B,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACA;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACA;AACA,sBAAsB,sDAAM;AAC5B,oBAAoB,cAAc;AAClC;AACA;AACA;AACO;AACP;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACO;AACP;AACA;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA,oBAAoB,gBAAgB;AACpC,wBAAwB,gBAAgB;AACxC,2BAA2B,oDAAW;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,gBAAgB;AAC3B,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,oBAAoB,YAAY;AAChC,4BAA4B,YAAY;AACxC;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,gBAAgB;AAClD;AACP;AACA;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB,WAAW,SAAS;AACpB,YAAY,UAAU;AACtB;AACO;AACP;AACA,oCAAoC;AACpC,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,oCAAoC;AAC/C;AACA,aAAa,UAAU;AACvB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kDAAS;AAC7B;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB;AACO;AACP;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB,WAAW,OAAO;AAClB,YAAY;AACZ;AACO;AACP;AACA;AACA;AACA,2CAA2C;;;;;;;;;;ACxO3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ,KAA0B;AAClC;AACA,MAAM,SAAS,IAA0C;AACzD,QAAQ,mCAAO,aAAa,gBAAgB;AAAA,kGAAC;AAC7C,MAAM,KAAK,EAEN;AACL,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA,qBAAqB,QAAQ;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC;AACxC,wCAAwC;AACxC,2CAA2C;AAC3C,+BAA+B;AAC/B,8BAA8B;AAC9B,+BAA+B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,wBAAwB;AAChD;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,mBAAmB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,uBAAuB;AAC/C;AACA;AACA;AACA;AACA;AACA,wBAAwB,uBAAuB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,wBAAwB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qDAAqD;AACrD;AACA,2BAA2B;AAC3B,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA,wBAAwB,iBAAiB;AACzC;AACA;AACA,4BAA4B,SAAS;AACrC;AACA;AACA;AACA,wBAAwB,gCAAgC;AACxD;AACA;AACA;AACA,oBAAoB,SAAS;AAC7B;AACA;AACA,oBAAoB,OAAO;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;;;;;;;;;;;AC1aY;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,OAAO;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;AACA,SAAS,eAAe;AACxB;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA;AACA,oBAAoB,OAAO;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,SAAS;AAC/B;AACA;AACA;AACA,kBAAkB,OAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,SAAS;AAC/B;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;AClJa;AACb;AACA,8CAA6C,EAAE,aAAa,EAAC;AAC7D,mBAAmB,GAAG,YAAY;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B,8CAA8C,qCAAqC;AACnF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;;;;;;;;;;;;;;;;;;;;;;;;;ACtFZ;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACA;AACA;AACA;AACA;AACA;AACA;AACP;AACO;AACA;;;;;;;;;;;;;;;;;ACtBP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB;AACxB,yBAAyB;AACzB,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,eAAe;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,eAAe;AACvC;AACA,4BAA4B,eAAe;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,4BAA4B,mBAAmB;AAC/C;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,4BAA4B,mBAAmB;AAC/C;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;AChfA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACnC;AACP;AACA;AACA,kCAAkC,6DAAmB;AACrD,wBAAwB,qBAAqB;AAC7C,4BAA4B,wBAAwB;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA,2DAA2D;AAC3D;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,uDAAuD;AACvD;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,qBAAqB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D;AAC/D;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C,+FAA+F;AAC/F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D;AAC/D;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iFAAiF;AACjF;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,gBAAgB;AACvC;AACA;AACA,yBAAyB,oBAAoB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8CAA8C;AAC9C,gDAAgD;AAChD;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACrUiC;AACK;AACsC;AACE;AACL;AACF;AACjC;AAC2D;AAClB;AAC/E;AACA,+BAA+B,gGAAsB;AACrD;AACA;AACA;AACA;AACA,mBAAmB,iGAA6B;AAChD;AACA,mBAAmB,qGAAgC;AACnD;AACA,mBAAmB,qGAAgC;AACnD;AACA,mBAAmB,qGAAgC;AACnD;AACA;AACA,mBAAmB,yFAAwB;AAC3C;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,iBAAiB,EAAE,IAAI;AAC/C;AACA,KAAK;AACL;AACA;AACO,gDAAgD,6DAAmB;AAC1E,iBAAiB;AACjB,qBAAqB;AACrB,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA,wCAAwC,sGAAyB;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,6FAAmB;AAChD;AACA,mIAAmI,yDAAe,CAAC,gDAAM,wCAAwC,yDAAe;AAChN;AACA;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA,wDAAwD,qDAAe;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kDAAkD,uDAAa;AAC/D;AACA;AACA;AACA,6BAA6B,6FAAmB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,4BAA4B,qCAAqC;AACjE;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,qCAAqC;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,8FAAiB;AACzC;AACA,wBAAwB,0FAAa;AACrC;AACA;AACA;AACA;AACA;AACA,iBAAiB,oGAAmB;AACpC;AACA,SAAS;AACT;AACA;AACA;AACA;AACO,kCAAkC,6DAAmB;AAC5D,iBAAiB,OAAO,yDAAmB;AAC3C,qBAAqB,OAAO,yDAAmB;AAC/C,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA,qEAAqE,qDAAe;AACpF;AACA;AACA;AACA;AACA;AACA;AACO,kDAAkD,6DAAmB;AAC5E,iBAAiB;AACjB,qBAAqB,OAAO,0EAAoC;AAChE,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oDAAoD,sDAAgB;AACpE,gDAAgD,uDAAa;AAC7D,yBAAyB,6FAAmB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,yFAAwB;AAC9C;AACA;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA,8BAA8B,oGAAmB;AACjD,8BAA8B,oGAAmB;AACjD;AACA;AACA;AACA,2BAA2B,oGAAmB;AAC9C;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACtQO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,sCAAsC;AAChC;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,oBAAoB;AACd;AACP;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,8BAA8B;AACxB;AACA;AACA;AACA;AACA;AACP;AACA;AACA,CAAC,sBAAsB;AAChB;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,YAAY,6GAA6G;AACzH,YAAY,wGAAwG;AACpH,WAAW,mGAAmG;AAC9G,YAAY,oEAAoE;AAChF;AACA;AACA;AACA,KAAK;AACL,WAAW,0DAA0D;AACrE;;;;;;;;;;;;;;;;;;;;;;;;ACtDA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACsC;AACI;AACjB;AACgE;AACzF;AACA;AACA;AACA;AACA;AACA,WAAW,WAAW;AACtB;AACO;AACP,0BAA0B,mGAAiB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,wDAAc;AAC9C,2BAA2B,wDAAc;AACzC;AACA;AACA;AACA,YAAY,+CAAC;AACb;AACA,YAAY,+CAAC;AACb;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,wBAAwB,mDAAS;AACjC,iBAAiB,gDAAM;AACvB,YAAY,oDAAU;AACtB;AACA;AACA;AACA;AACA;AACA;AAC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C,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACsC;AAC2D;AAC1F;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA,aAAa;AACb,0CAA0C,+GAAoC;AAC9E,qEAAqE,oEAA0B;AAC/F,iBAAiB,yDAAyD,uDAAa;AACvF,KAAK;AACL;AACO;AACP;AACA;AACA,+BAA+B,GAAG,GAAG,WAAW;AAChD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACnCsC;AACiC;AAC7B;AACgK;AAC3H;AACxE;AACA;AACA;AACA;AACA;AACP,uBAAuB,sDAAkB;AACzC;AACA,2BAA2B,uDAAa;AACxC,2BAA2B,wGAA2B;AACtD,yBAAyB,6FAAmB;AAC5C;AACA,oBAAoB,gBAAgB;AACpC;AACA;AACA;AACA;AACA,mCAAmC,sDAAkB;AACrD,oBAAoB,0DAAgB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,WAAW,+DAAqB;AAChC;AACO;AACP,2BAA2B,uDAAa;AACxC;AACA,yBAAyB,6FAAmB;AAC5C;AACA;AACA,oBAAoB,gBAAgB;AACpC;AACA;AACA;AACA,wBAAwB,qBAAqB;AAC7C;AACA;AACA,oBAAoB,4DAAkB,YAAY,aAAa;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,mCAAmC,uDAAa;AAChD;AACA,yBAAyB,6FAAmB;AAC5C;AACA;AACA;AACA;AACA;AACA,oBAAoB,qBAAqB;AACzC;AACA;AACA,gBAAgB,4DAAkB,YAAY,aAAa;AAC3D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,KAAK;AACL;AACA;AACO;AACP;AACA,oBAAoB,oBAAoB;AACxC;AACA,oBAAoB,yDAAqB;AACzC,wBAAwB,kDAAc;AACtC,gDAAgD,yDAAqB;AACrE,iEAAiE,oDAAgB;AACjF;AACA,6BAA6B,gDAAY;AACzC,0CAA0C,yDAAqB;AAC/D;AACA;AACA;AACA;AACA;AACA,2BAA2B,oDAAgB,4BAA4B,WAAW,eAAe,WAAW;AAC5G,2BAA2B,mDAAe,0BAA0B,WAAW,UAAU,WAAW;AACpG,2BAA2B,kDAAc;AACzC,2BAA2B,gDAAY,QAAQ,WAAW;AAC1D;AACA,iBAAiB;AACjB;AACA;AACA;AACA,oBAAoB,yDAAqB;AACzC,6CAA6C,yDAAqB;AAClE;AACA,SAAS;AACT;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;AC3HA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACsC;AACA;AACgC;AACtE,YAAY,gFAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,qDAAW;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE,6DAA6D;AACpI,sEAAsE,6BAA6B;AACnG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,6FAA6F;AACjH,oBAAoB,6FAA6F;AACjH;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB;AACA;AACA,2BAA2B,0DAAgB;AAC3C;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,iCAAiC;AACjC,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD,4BAA4B;AAC5E,kDAAkD,6BAA6B;AAC/E;AACA,qEAAqE,4CAA4C;AACjH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,qEAAqE,0DAAgB;AACrF;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE,qCAAqC;AAC5G;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,6CAA6C,mCAAmC,YAAY,OAAO,EAAE,+BAA+B,GAAG;AACvI;AACA;AACA;AACA,iCAAiC,2BAA2B;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,kDAAQ;AACtC;AACA;AACA,gDAAgD,gDAAM;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,uCAAuC,gDAAM,gBAAgB,OAAO,EAAE,oBAAoB;AAC1F;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA,qBAAqB;AACrB;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,kCAAkC,4CAA4C;AAC9E,wBAAwB,gDAAM,oBAAoB,UAAU,uBAAuB,GAAG;AACtF;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,6BAA6B,4BAA4B;AACzD,iCAAiC,6BAA6B;AAC9D;AACA,iEAAiE,UAAU;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA,+CAA+C,IAAI,OAAO,KAAK;AAC/D;AACA;;;;;;;;;;;;;;;;;;;;;;;;ACjQA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AACoB;AACN;AACpD;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,OAAO;AAClB,YAAY,oBAAoB;AAChC;AACO;AACP;AACA,mBAAmB,4EAAkC;AACrD;AACA;AACA;AACA;AACA,oCAAoC,sEAAwB;AAC5D,qCAAqC,gEAAW;AAChD;AACA,oCAAoC,6DAAmB,gBAAgB,YAAY;AACnF;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,oBAAoB;AAChF,gCAAgC,gDAAM,OAAO,6CAA6C;AAC1F;AACA,uDAAuD,0BAA0B;AACjF;AACA,6CAA6C,YAAY;AACzD;AACA;AACA;AACA,iBAAiB;AACjB,+BAA+B,0DAAgB;AAC/C,8BAA8B,iDAAO;AACrC,6BAA6B,gDAAM;AACnC;AACA;AACA,6BAA6B,oDAAU;AACvC;AACA;AACA;AACA,yBAAyB,oDAAU;AACnC;AACA;AACA,mBAAmB,mDAAS;AAC5B,KAAK;AACL;;;;;;;;;;;;AC/DA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;UAEA;UACA;;;;;WCzBA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;WACA;WACA;WACA;WACA;;;;;WCJA;WACA;WACA;WACA;WACA,GAAG;WACH;WACA;WACA,CAAC;;;;;WCPD;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;WCNA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;;;;;WCfA;;WAEA;WACA;WACA;WACA;WACA;WACA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACrBA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACA;AAC0C;AACJ;AACA;AAC/B,qBAAqB,oDAAU;AAC2D;AAC1B;AACT;AACE;AAChB;AAC8B;AACS;AAC+B;AACE;AACrE;AACuB;AAChC;AACwB;AACa;AACI;AACG;AACH;AAC5C;AACvC;AACO;AACP;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,mFAAiC;AAChD;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,mFAAiC;AAChD;AACA;AACA;AACA;AACA,QAAQ,4DAAkB;AAC1B;AACA;AACO;AACP;AACA;AACA,mBAAmB,0FAAY;AAC/B,wBAAwB,kEAAwB;AAChD,QAAQ,4DAAkB;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,wCAAwC,EAAE,gCAAgC,EAAE,gBAAgB;AACxH,qBAAqB,MAAM,2DAA2D,YAAY;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,wCAAwC,EAAE,gCAAgC,EAAE,eAAe;AAC3H,yBAAyB,MAAM,2DAA2D,YAAY;AACtG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACO;AACP,wBAAwB,yDAAiB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,iFAAO;AACtB;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,uEAAe;AAC9B;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA;AACA,6BAA6B;AACtB;AACP;AACA;AACA;AACA,2BAA2B,6DAAc;AACzC;AACA;AACA,0BAA0B,4EAAqB;AAC/C;AACA,qBAAqB,4CAA4C;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,qGAAiB,iHAAiH,kEAAwB,QAAQ,gEAAa,EAAE,kFAAoB,EAAE,kFAAoB,EAAE,oFAAsB;AAC5Q;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,gCAAgC;AAChC,6BAA6B;AAC7B,mCAAmC;AACnC;AACO;AACP;AACA;AACA;AACA,2BAA2B,6DAAc;AACzC;AACA;AACA,+BAA+B,4EAAqB;AACpD,mCAAmC,kEAAwB;AAC3D,8BAA8B,qGAAqB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,oEAAa;AACpD;AACA;AACA;AACA;AACA,8BAA8B,4DAAkB;AAChD;AACA;AACA;AACA,4BAA4B,0DAAgB;AAC5C;AACA,yCAAyC,qEAAqE;AAC9G;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AACxB;AACP;AACA,YAAY,sDAAY,GAAG,yCAAyC;AACpE,YAAY,4DAAkB;AAC9B;AACA;AACA;AACA;AACA,gEAAgE,iEAAsB;AACtF;AACA,4BAA4B,iEAAkB;AAC9C,6BAA6B,iEAAW;AACxC,oBAAoB,+DAAqB;AACzC,sBAAsB,6DAAmB;AACzC,iBAAiB,uDAAa;AAC9B;AACA,cAAc,uEAA6B;AAC3C,KAAK;AACL;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACO;AACP;AACA,qBAAqB,sDAAY,GAAG,sDAAsD;AAC1F;AACA;AACA,8DAA8D,SAAS;AACvE,6BAA6B,6EAAS;AACtC;AACA;AACA,cAAc,uEAA6B;AAC3C;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,2BAA2B;AAC3B;AACO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACO;AACP;AACA;AACA,mBAAmB,uDAAa;AAChC;AACA;AACA;AACA,+BAA+B,kEAAwB;AACvD;AACA,8BAA8B,0FAAY;AAC1C;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,uDAAuD,8BAA8B;AACrF,YAAY,wEAA8B,WAAW,2DAAiB;AACtE,SAAS;AACT;AACA;AACA,YAAY,0DAAgB;AAC5B;AACA;AACA;AACA;AACA,6BAA6B,wDAAc;AAC3C,YAAY,mDAAS;AACrB;AACA;AACA,aAAa;AACb,qBAAqB,gDAAM;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACO;AACP;AACA,mCAAmC,0EAA2B;AAC9D,KAAK;AACL;AACA;AACA;AACA,6BAA6B;AAC7B;AACO;AACP;AACA,gEAAgE,iEAAsB;AACtF;AACA,eAAe,gFAAqB;AACpC,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,oBAAoB,8FAAgB;AACpC;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACd;AACP,IAAI,wDAAO;AACX;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,qEAAmB;AAClC;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,qFAAmC;AAClD;AACA;AACA,sBAAsB;AACtB;AACO;AACP;AACA,mBAAmB,4EAAkC;AACrD,+BAA+B,8DAAoB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,oEAA0B;AAC5D,2BAA2B,8DAAoB;AAC/C;AACA,0CAA0C,6DAAmB,8BAA8B,UAAU;AACrG,oCAAoC,kEAAwB;AAC5D,gDAAgD,cAAc,YAAY,SAAS;AACnF,wCAAwC,QAAQ,WAAW,0BAA0B;AACrF;AACA;AACA;AACA;AACA,+CAA+C,WAAW,uDAAa,EAAE;AACzE,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,4EAA4E,kBAAkB;AAC9F;AACA;AACA,QAAQ,yDAAe;AACvB;AACA,sBAAsB,kEAAwB;AAC9C,qDAAqD,KAAK;AAC1D;AACA,KAAK;AACL;AACA;AACA;AACA,qBAAqB;AACd;AACP;AACA,oBAAoB,uGAAyB,WAAW,kDAAK;AAC7D,eAAe,0DAAgB;AAC/B,mBAAmB,kGAAqB;AACxC;AACA;AACA;AACA,yBAAyB,gEAAmB;AAC5C,sBAAsB,+DAAqB,EAAE,gEAAmB;AAChE,sBAAsB,4DAAe,aAAa,4DAAe;AACjE;AACA,UAAU,sDAAY;AACtB;AACA;AACA,qBAAqB;AACrB;AACO;AACP,kBAAkB,0FAAgB,YAAY,gGAAsB;AACpE;AACA","sources":["webpack://bio/./node_modules/@biowasm/aioli/dist/aioli.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/aminoacids.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/nucleotides.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/seq-palettes.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/sequence-encoder.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/unknown.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/cell-renderer.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/fasta-handler.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/notation-converter.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/splitter.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/units-handler.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/vd-regions.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/viewers/web-logo.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/sequence-space.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/typed-metrics.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/viewers/activity-cliffs.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/workers/dimensionality-reducing-worker-creator.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/dataframe-utils.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/random.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/similarity-metrics.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/type-declarations.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/vector-operations.js","webpack://bio/./node_modules/fast-sha256/sha256.js","webpack://bio/./node_modules/fastest-levenshtein/index.js","webpack://bio/./node_modules/jaro-winkler-typescript/lib/index.js","webpack://bio/./src/const.ts","webpack://bio/./src/seq_align.ts","webpack://bio/./src/utils/atomic-works.ts","webpack://bio/./src/utils/cell-renderer.ts","webpack://bio/./src/utils/constants.ts","webpack://bio/./src/utils/convert.ts","webpack://bio/./src/utils/multiple-sequence-alignment.ts","webpack://bio/./src/utils/sequence-activity-cliffs.ts","webpack://bio/./src/utils/sequence-space.ts","webpack://bio/./src/utils/utils.ts","webpack://bio/./src/viewers/vd-regions-viewer.ts","webpack://bio/./src/widgets/representations.ts","webpack://bio/external var \"$\"","webpack://bio/external var \"DG\"","webpack://bio/external var \"grok\"","webpack://bio/external var \"rxjs\"","webpack://bio/external var \"ui\"","webpack://bio/external var \"wu\"","webpack://bio/webpack/bootstrap","webpack://bio/webpack/runtime/compat get default export","webpack://bio/webpack/runtime/define property getters","webpack://bio/webpack/runtime/get javascript chunk filename","webpack://bio/webpack/runtime/global","webpack://bio/webpack/runtime/hasOwnProperty shorthand","webpack://bio/webpack/runtime/make namespace object","webpack://bio/webpack/runtime/publicPath","webpack://bio/webpack/runtime/jsonp chunk loading","webpack://bio/./src/package.ts"],"sourcesContent":["(function (global, factory) {\n  typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n  typeof define === 'function' && define.amd ? define(factory) :\n  (global = global || self, global.Aioli = factory());\n}(this, (function () { 'use strict';\n\n  var name = \"@biowasm/aioli\";\n  var version = \"2.4.0-rc2\";\n  var description = \"A framework for building WebAssembly-based genomics tools\";\n  var browser = \"dist/aioli.js\";\n  var worker = \"dist/aioli.worker.js\";\n  var main = \"aioli.js\";\n  var repository = {\n  \ttype: \"git\",\n  \turl: \"git+https://github.com/biowasm/aioli.git\"\n  };\n  var author = \"Robert Aboukhalil\";\n  var license = \"MIT\";\n  var dependencies = {\n  \tcomlink: \"^4.3.1\",\n  \t\"wasm-feature-detect\": \"^1.2.11\"\n  };\n  var devDependencies = {\n  \t\"@rollup/plugin-commonjs\": \"^11.0.1\",\n  \t\"@rollup/plugin-json\": \"^4.1.0\",\n  \t\"@rollup/plugin-node-resolve\": \"^7.0.0\",\n  \tcypress: \"^8.0.0\",\n  \trollup: \"^1.29.0\",\n  \t\"rollup-plugin-terser\": \"^7.0.2\"\n  };\n  var scripts = {\n  \tbuild: \"rollup -c\",\n  \tdev: \"rollup -c -w\",\n  \ttest: \"cypress run\",\n  \t\"test-open\": \"cypress open\"\n  };\n  var files = [\n  \t\"dist\"\n  ];\n  var pkg = {\n  \tname: name,\n  \tversion: version,\n  \tdescription: description,\n  \tbrowser: browser,\n  \tworker: worker,\n  \tmain: main,\n  \trepository: repository,\n  \tauthor: author,\n  \tlicense: license,\n  \tdependencies: dependencies,\n  \tdevDependencies: devDependencies,\n  \tscripts: scripts,\n  \tfiles: files\n  };\n\n  /**\r\n   * Copyright 2019 Google Inc. All Rights Reserved.\r\n   * Licensed under the Apache License, Version 2.0 (the \"License\");\r\n   * you may not use this file except in compliance with the License.\r\n   * You may obtain a copy of the License at\r\n   *     http://www.apache.org/licenses/LICENSE-2.0\r\n   * Unless required by applicable law or agreed to in writing, software\r\n   * distributed under the License is distributed on an \"AS IS\" BASIS,\r\n   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n   * See the License for the specific language governing permissions and\r\n   * limitations under the License.\r\n   */\r\n  const proxyMarker = Symbol(\"Comlink.proxy\");\r\n  const createEndpoint = Symbol(\"Comlink.endpoint\");\r\n  const releaseProxy = Symbol(\"Comlink.releaseProxy\");\r\n  const throwMarker = Symbol(\"Comlink.thrown\");\r\n  const isObject = (val) => (typeof val === \"object\" && val !== null) || typeof val === \"function\";\r\n  /**\r\n   * Internal transfer handle to handle objects marked to proxy.\r\n   */\r\n  const proxyTransferHandler = {\r\n      canHandle: (val) => isObject(val) && val[proxyMarker],\r\n      serialize(obj) {\r\n          const { port1, port2 } = new MessageChannel();\r\n          expose(obj, port1);\r\n          return [port2, [port2]];\r\n      },\r\n      deserialize(port) {\r\n          port.start();\r\n          return wrap(port);\r\n      },\r\n  };\r\n  /**\r\n   * Internal transfer handler to handle thrown exceptions.\r\n   */\r\n  const throwTransferHandler = {\r\n      canHandle: (value) => isObject(value) && throwMarker in value,\r\n      serialize({ value }) {\r\n          let serialized;\r\n          if (value instanceof Error) {\r\n              serialized = {\r\n                  isError: true,\r\n                  value: {\r\n                      message: value.message,\r\n                      name: value.name,\r\n                      stack: value.stack,\r\n                  },\r\n              };\r\n          }\r\n          else {\r\n              serialized = { isError: false, value };\r\n          }\r\n          return [serialized, []];\r\n      },\r\n      deserialize(serialized) {\r\n          if (serialized.isError) {\r\n              throw Object.assign(new Error(serialized.value.message), serialized.value);\r\n          }\r\n          throw serialized.value;\r\n      },\r\n  };\r\n  /**\r\n   * Allows customizing the serialization of certain values.\r\n   */\r\n  const transferHandlers = new Map([\r\n      [\"proxy\", proxyTransferHandler],\r\n      [\"throw\", throwTransferHandler],\r\n  ]);\r\n  function expose(obj, ep = self) {\r\n      ep.addEventListener(\"message\", function callback(ev) {\r\n          if (!ev || !ev.data) {\r\n              return;\r\n          }\r\n          const { id, type, path } = Object.assign({ path: [] }, ev.data);\r\n          const argumentList = (ev.data.argumentList || []).map(fromWireValue);\r\n          let returnValue;\r\n          try {\r\n              const parent = path.slice(0, -1).reduce((obj, prop) => obj[prop], obj);\r\n              const rawValue = path.reduce((obj, prop) => obj[prop], obj);\r\n              switch (type) {\r\n                  case \"GET\" /* GET */:\r\n                      {\r\n                          returnValue = rawValue;\r\n                      }\r\n                      break;\r\n                  case \"SET\" /* SET */:\r\n                      {\r\n                          parent[path.slice(-1)[0]] = fromWireValue(ev.data.value);\r\n                          returnValue = true;\r\n                      }\r\n                      break;\r\n                  case \"APPLY\" /* APPLY */:\r\n                      {\r\n                          returnValue = rawValue.apply(parent, argumentList);\r\n                      }\r\n                      break;\r\n                  case \"CONSTRUCT\" /* CONSTRUCT */:\r\n                      {\r\n                          const value = new rawValue(...argumentList);\r\n                          returnValue = proxy(value);\r\n                      }\r\n                      break;\r\n                  case \"ENDPOINT\" /* ENDPOINT */:\r\n                      {\r\n                          const { port1, port2 } = new MessageChannel();\r\n                          expose(obj, port2);\r\n                          returnValue = transfer(port1, [port1]);\r\n                      }\r\n                      break;\r\n                  case \"RELEASE\" /* RELEASE */:\r\n                      {\r\n                          returnValue = undefined;\r\n                      }\r\n                      break;\r\n                  default:\r\n                      return;\r\n              }\r\n          }\r\n          catch (value) {\r\n              returnValue = { value, [throwMarker]: 0 };\r\n          }\r\n          Promise.resolve(returnValue)\r\n              .catch((value) => {\r\n              return { value, [throwMarker]: 0 };\r\n          })\r\n              .then((returnValue) => {\r\n              const [wireValue, transferables] = toWireValue(returnValue);\r\n              ep.postMessage(Object.assign(Object.assign({}, wireValue), { id }), transferables);\r\n              if (type === \"RELEASE\" /* RELEASE */) {\r\n                  // detach and deactive after sending release response above.\r\n                  ep.removeEventListener(\"message\", callback);\r\n                  closeEndPoint(ep);\r\n              }\r\n          });\r\n      });\r\n      if (ep.start) {\r\n          ep.start();\r\n      }\r\n  }\r\n  function isMessagePort(endpoint) {\r\n      return endpoint.constructor.name === \"MessagePort\";\r\n  }\r\n  function closeEndPoint(endpoint) {\r\n      if (isMessagePort(endpoint))\r\n          endpoint.close();\r\n  }\r\n  function wrap(ep, target) {\r\n      return createProxy(ep, [], target);\r\n  }\r\n  function throwIfProxyReleased(isReleased) {\r\n      if (isReleased) {\r\n          throw new Error(\"Proxy has been released and is not useable\");\r\n      }\r\n  }\r\n  function createProxy(ep, path = [], target = function () { }) {\r\n      let isProxyReleased = false;\r\n      const proxy = new Proxy(target, {\r\n          get(_target, prop) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              if (prop === releaseProxy) {\r\n                  return () => {\r\n                      return requestResponseMessage(ep, {\r\n                          type: \"RELEASE\" /* RELEASE */,\r\n                          path: path.map((p) => p.toString()),\r\n                      }).then(() => {\r\n                          closeEndPoint(ep);\r\n                          isProxyReleased = true;\r\n                      });\r\n                  };\r\n              }\r\n              if (prop === \"then\") {\r\n                  if (path.length === 0) {\r\n                      return { then: () => proxy };\r\n                  }\r\n                  const r = requestResponseMessage(ep, {\r\n                      type: \"GET\" /* GET */,\r\n                      path: path.map((p) => p.toString()),\r\n                  }).then(fromWireValue);\r\n                  return r.then.bind(r);\r\n              }\r\n              return createProxy(ep, [...path, prop]);\r\n          },\r\n          set(_target, prop, rawValue) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              // FIXME: ES6 Proxy Handler `set` methods are supposed to return a\r\n              // boolean. To show good will, we return true asynchronously ¯\\_(ツ)_/¯\r\n              const [value, transferables] = toWireValue(rawValue);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"SET\" /* SET */,\r\n                  path: [...path, prop].map((p) => p.toString()),\r\n                  value,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n          apply(_target, _thisArg, rawArgumentList) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              const last = path[path.length - 1];\r\n              if (last === createEndpoint) {\r\n                  return requestResponseMessage(ep, {\r\n                      type: \"ENDPOINT\" /* ENDPOINT */,\r\n                  }).then(fromWireValue);\r\n              }\r\n              // We just pretend that `bind()` didn’t happen.\r\n              if (last === \"bind\") {\r\n                  return createProxy(ep, path.slice(0, -1));\r\n              }\r\n              const [argumentList, transferables] = processArguments(rawArgumentList);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"APPLY\" /* APPLY */,\r\n                  path: path.map((p) => p.toString()),\r\n                  argumentList,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n          construct(_target, rawArgumentList) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              const [argumentList, transferables] = processArguments(rawArgumentList);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"CONSTRUCT\" /* CONSTRUCT */,\r\n                  path: path.map((p) => p.toString()),\r\n                  argumentList,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n      });\r\n      return proxy;\r\n  }\r\n  function myFlat(arr) {\r\n      return Array.prototype.concat.apply([], arr);\r\n  }\r\n  function processArguments(argumentList) {\r\n      const processed = argumentList.map(toWireValue);\r\n      return [processed.map((v) => v[0]), myFlat(processed.map((v) => v[1]))];\r\n  }\r\n  const transferCache = new WeakMap();\r\n  function transfer(obj, transfers) {\r\n      transferCache.set(obj, transfers);\r\n      return obj;\r\n  }\r\n  function proxy(obj) {\r\n      return Object.assign(obj, { [proxyMarker]: true });\r\n  }\r\n  function toWireValue(value) {\r\n      for (const [name, handler] of transferHandlers) {\r\n          if (handler.canHandle(value)) {\r\n              const [serializedValue, transferables] = handler.serialize(value);\r\n              return [\r\n                  {\r\n                      type: \"HANDLER\" /* HANDLER */,\r\n                      name,\r\n                      value: serializedValue,\r\n                  },\r\n                  transferables,\r\n              ];\r\n          }\r\n      }\r\n      return [\r\n          {\r\n              type: \"RAW\" /* RAW */,\r\n              value,\r\n          },\r\n          transferCache.get(value) || [],\r\n      ];\r\n  }\r\n  function fromWireValue(value) {\r\n      switch (value.type) {\r\n          case \"HANDLER\" /* HANDLER */:\r\n              return transferHandlers.get(value.name).deserialize(value.value);\r\n          case \"RAW\" /* RAW */:\r\n              return value.value;\r\n      }\r\n  }\r\n  function requestResponseMessage(ep, msg, transfers) {\r\n      return new Promise((resolve) => {\r\n          const id = generateUUID();\r\n          ep.addEventListener(\"message\", function l(ev) {\r\n              if (!ev.data || !ev.data.id || ev.data.id !== id) {\r\n                  return;\r\n              }\r\n              ep.removeEventListener(\"message\", l);\r\n              resolve(ev.data);\r\n          });\r\n          if (ep.start) {\r\n              ep.start();\r\n          }\r\n          ep.postMessage(Object.assign({ id }, msg), transfers);\r\n      });\r\n  }\r\n  function generateUUID() {\r\n      return new Array(4)\r\n          .fill(0)\r\n          .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16))\r\n          .join(\"-\");\r\n  }\n  //# sourceMappingURL=comlink.mjs.map\n\n  // Constants\n  const URL_CDN_ROOT = \"https://cdn.biowasm.com/v2\";\n  const CONFIG_DEFAULTS = {\n  \t// Biowasm CDN URLs\n  \turlCDN: URL_CDN_ROOT,\n  \t// Get the Worker code corresponding to the current Aioli version\n  \turlAioli: `${URL_CDN_ROOT}/aioli/${pkg.version}/aioli.worker.js`,\n  \t// Where we can find the base biowasm module (only modify this for local development)\n  \turlBaseModule: null,\n\n  \t// Folder to use for mounting the shared filesystem\n  \tdirShared: \"/shared\",\n  \t// Folder to use for mounting File/Blob objects to the virtual file system\n  \tdirMounted: \"/mnt\",\n  \t// Folder to use for symlinks (basically, we make a symlink to each file mounted on WORKERFS\n  \t// so that operations like \"samtools index\" don't crash due to the read-only nature of WORKERS).\n  \t// Also mount URLs lazily in that folder.\n  \tdirData: \"/data\",\n  \t// Interleave stdout/stderr. If set to false, `.exec()` returns an object { \"stdout\": <text>, \"stderr\": <text> }\n  \tprintInterleaved: true,\n\n  \t// Callback function to run whenever we receive a message from the WebWorker with payload { type: \"biowasm\", value: ... }.\n  \t// See <https://github.com/biowasm/biowasm/tree/main/tools/bhtsne> for an example of how this can be used to send regular updates\n  \t// back to the main thread before callMain() is done running.\n  \tcallback: null,\n\n  \t// Debugging\n  \tdebug: false,\n  \tenv: \"prd\"\n  };\n\n  // Class: 1 object = 1 worker; user can decide if they want tools running in separate threads or all of them in one\n  class Aioli\n  {\n  \tconstructor(tools, config={})\n  \t{\n  \t\tif(tools == null)\n  \t\t\tthrow \"Expecting array of tools as input to Aioli constructor.\";\n\n  \t\t// Parse user input\n  \t\tif(!Array.isArray(tools))\n  \t\t\ttools = [ tools ];\n  \t\t// Overwrite default config if specified\n  \t\tconfig = Object.assign({}, CONFIG_DEFAULTS, config);\n  \t\t// For convenience, support \"<tool>/<version>\" or \"<tool>/<program>/<version>\" instead of object config\n  \t\ttools = tools.map(this._parseTool);\n  \t\t// If testing with different environment e.g. cdn-stg.biowasm.com\n  \t\tif(config.env != \"prd\") {\n  \t\t\tconfig.urlCDN = config.urlCDN.replace(\"cdn\", `cdn-${config.env}`);\n  \t\t\tconfig.urlAioli = config.urlAioli.replace(\"cdn\", `cdn-${config.env}`);\n  \t\t}\n\n  \t\t// Add biowasm base module to list of tools to initialize (need this for the shared virtual filesystem)\n  \t\ttools = [{\n  \t\t\ttool: \"base\",\n  \t\t\tversion: pkg.version,\n  \t\t\turlPrefix: config.urlBaseModule\n  \t\t}, ...tools];\n\n  \t\t// Set state\n  \t\tthis.tools = tools;\n  \t\tthis.config = config;\n\n  \t\t// Handle callback (delete it because we can't send a function to the WebWorker)\n  \t\tif(this.config.callback != null)\n  \t\t\tthis.callback = this.config.callback;\n  \t\tdelete this.config.callback;\n\n  \t\treturn this.init();\n  \t}\n\n  \t// Initialize the WebWorker and the WebAssembly modules within it\n  \tasync init() {\n  \t\t// Note: We can only create a WebWorker using a URL that has the same origin as the app running it, i.e. we\n  \t\t// can't do `new Worker(\"https://cdn.biowasm.com/v2/...\")`, so instead we need to fetch the contents of that\n  \t\t// URL first, and then create the WebWorker. See <https://stackoverflow.com/a/60015898>.\n  \t\tconst moduleJS = await (await fetch(this.config.urlAioli)).text();\n  \t\tconst moduleBlob = new Blob([moduleJS], { type: \"application/javascript\" });\n\n  \t\t// Create the WebWorker\n  \t\tconst worker = new Worker(URL.createObjectURL(moduleBlob));\n  \t\t// Listen for \"biowasm\" messages from the WebWorker\n  \t\tif(this.callback)\n  \t\t\tworker.onmessage = e => {\n  \t\t\t\tif(e.data.type == \"biowasm\")\n  \t\t\t\t\tthis.callback(e.data.value);\n  \t\t\t};\n\n  \t\tconst aioli = wrap(worker);\n  \t\taioli.tools = this.tools;\n  \t\taioli.config = this.config;\n\n  \t\t// Initialize the tools inside the WebWorker\n  \t\tawait aioli.init();\n\n  \t\treturn aioli;\n  \t}\n\n  \t// Parse \"<tool>/<version>\" and \"<tool>/<program>/<version>\" into { \"tool\": <tool>, \"program\": <program>, \"version\": <version> }\n  \t_parseTool(tool)\n  \t{\n  \t\t// If not a string, leave it as is\n  \t\tif(typeof tool !== \"string\")\n  \t\t\treturn tool;\n\n  \t\t// Support \"<tool>/<version>\" and \"<tool>/<program>/<version>\"\n  \t\tconst toolSplit = tool.split(\"/\");\n  \t\tif(toolSplit.length != 2 && toolSplit.length != 3)\n  \t\t\tthrow \"Expecting '<tool>/<version>' or '<tool>/<program>/<version>'\";\n\n  \t\treturn {\n  \t\t\ttool: toolSplit[0],\n  \t\t\tprogram: toolSplit.length == 3 ? toolSplit[1] : toolSplit[0],\n  \t\t\tversion: toolSplit[toolSplit.length - 1]\n  \t\t};\n  \t}\n  }\n\n  return Aioli;\n\n})));\n//# sourceMappingURL=aioli.js.map\n","import { SeqPaletteBase } from './seq-palettes';\nexport class AminoacidsPalettes extends SeqPaletteBase {\n    static get Lesk() {\n        if (this.lesk === void 0) {\n            this.lesk = this.makePalette([\n                [['G', 'A', 'S', 'T'], 'orange'],\n                [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],\n                [['N', 'Q', 'H'], 'magenta'],\n                [['D', 'E'], 'red'],\n                [['K', 'R'], 'all_blue'],\n            ], false, AminoacidsPalettes);\n        }\n        return this.lesk;\n    }\n    static get GrokGroups() {\n        if (this.grokGroups === void 0) {\n            this.grokGroups = this.makePalette([\n                [['C', 'U'], 'yellow'],\n                [['G', 'P'], 'red'],\n                [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],\n                [['R', 'H', 'K'], 'light_blue'],\n                [['D', 'E'], 'dark_blue'],\n                [['S', 'T', 'N', 'Q'], 'orange'],\n            ], false, AminoacidsPalettes);\n        }\n        return this.grokGroups;\n    }\n    static get RasMol() {\n        if (this.rasMol === void 0) {\n            this.rasMol = new AminoacidsPalettes({\n                // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm\n                'D': '#E60A0A',\n                'E': '#E60A0A',\n                'C': '#E6E600',\n                'M': '#E6E600',\n                'K': '#145AFF',\n                'R': '#145AFF',\n                'S': '#FA9600',\n                'T': '#FA9600',\n                'F': '#3232AA',\n                'Y': '#3232AA',\n                'N': '#00DCDC',\n                'Q': '#00DCDC',\n                'G': '#EBEBEB',\n                'L': '#0F820F',\n                'V': '#0F820F',\n                'I': '#0F820F',\n                'A': '#C8C8C8',\n                'W': '#B45AB4',\n                'H': '#8282D2',\n                'P': '#DC9682',\n                'others': '#BEA06E',\n            });\n        }\n        return this.rasMol;\n    }\n}\nexport class Aminoacids {\n    static getPalette(scheme = 'grok') {\n        switch (scheme) {\n            case 'grok':\n                return AminoacidsPalettes.GrokGroups;\n            case 'lesk':\n                return AminoacidsPalettes.Lesk;\n            default:\n                throw new Error(`ChemPalette: scheme \\`${scheme}\\` does not exist`);\n        }\n    }\n    /**\n     * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted\n     *\n     * @param {string} c raw amino\n     * @return {[string, string]} outer and inner content\n     */\n    static getInnerOuter(c) {\n        let isInner = 0;\n        let inner = '';\n        let outer = '';\n        for (const char of c) {\n            if (char == '(')\n                isInner++;\n            else if (char == ')')\n                isInner--;\n            else if (isInner)\n                inner += char;\n            else\n                outer += char;\n        }\n        return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];\n    }\n    static getColorAAPivot(monomer = '', scheme = 'grok') {\n        var _a, _b, _c;\n        //const chemPaletteInstance = AAPalettes.GrokGroups();\n        const chemPaletteInstance = this.getPalette(scheme);\n        let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);\n        outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);\n        innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);\n        if (monomer.length == 1 || monomer[1] == '(') {\n            const amino = (_a = monomer[0]) === null || _a === void 0 ? void 0 : _a.toUpperCase();\n            return amino in chemPaletteInstance ?\n                [chemPaletteInstance.get(amino), amino, innerMonomer, 1] :\n                [this.undefinedColor, outerMonomer, innerMonomer, 1];\n        }\n        if (monomer[0] == 'd' && monomer[1] in chemPaletteInstance) {\n            if (monomer.length == 2 || monomer[2] == '(') {\n                const amino = (_b = monomer[1]) === null || _b === void 0 ? void 0 : _b.toUpperCase();\n                return amino in chemPaletteInstance ?\n                    [chemPaletteInstance.get(amino), amino, innerMonomer, 2] :\n                    [this.undefinedColor, outerMonomer, innerMonomer, 2];\n            }\n        }\n        if (monomer.substring(0, 3) in this.AAFullNames) {\n            if (monomer.length == 3 || monomer[3] == '(') {\n                const amino = this.AAFullNames[monomer.substring(0, 3)];\n                return amino in chemPaletteInstance ?\n                    [chemPaletteInstance.get(amino), amino, innerMonomer, 3] :\n                    [this.undefinedColor, outerMonomer, innerMonomer, 3];\n            }\n        }\n        if (((_c = monomer[0]) === null || _c === void 0 ? void 0 : _c.toLowerCase()) == monomer[0]) {\n            if (monomer.substring(1, 3) in this.AAFullNames) {\n                if (monomer.length == 4 || monomer[4] == '(') {\n                    const amino = this.AAFullNames[monomer.substring(1, 3)];\n                    return amino in chemPaletteInstance ?\n                        [chemPaletteInstance.get(amino), amino, innerMonomer, 4] :\n                        [this.undefinedColor, outerMonomer, innerMonomer, 4];\n                }\n            }\n        }\n        return [this.undefinedColor, outerMonomer, innerMonomer, 0];\n    }\n}\nAminoacids.SemType = 'Aminoacids';\nAminoacids.SemTypeMultipleAlignment = 'AminoacidsMultipleAlignment';\nAminoacids.undefinedColor = 'rgb(100,100,100)';\nAminoacids.Names = {\n    'G': 'Glycine',\n    'L': 'Leucine',\n    'Y': 'Tyrosine',\n    'S': 'Serine',\n    'E': 'Glutamic acid',\n    'Q': 'Glutamine',\n    'D': 'Aspartic acid',\n    'N': 'Asparagine',\n    'F': 'Phenylalanine',\n    'A': 'Alanine',\n    'K': 'Lysine',\n    'R': 'Arginine',\n    'H': 'Histidine',\n    'C': 'Cysteine',\n    'V': 'Valine',\n    'P': 'Proline',\n    'W': 'Tryptophan',\n    'I': 'Isoleucine',\n    'M': 'Methionine',\n    'T': 'Threonine',\n};\nAminoacids.AASmiles = {\n    'G': 'NCC(=O)O',\n    'L': 'N[C@H](CC(C)C)C(=O)O',\n    'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',\n    'S': 'NC(CO)C(=O)O',\n    'E': 'N[C@@H](CCC(O)=O)C(=O)O',\n    'Q': 'N[C@@H](CCC(N)=O)C(=O)O',\n    'D': 'N[C@@H](CC(O)=O)C(=O)O',\n    'N': 'N[C@@H](CC(N)=O)C(=O)O',\n    'F': 'NC(CC1=CC=CC=C1)C(=O)O',\n    'A': 'N[C@H](C)C(=O)O',\n    'K': 'NC(CCCCN)C(=O)O',\n    'R': 'N[C@H](CCCNC(=N)C)C(=O)O',\n    'H': 'NC(CC1=CN=C[N]1)C(=O)O',\n    'C': 'N[C@@H](CS)C(=O)O',\n    'V': 'NC(C(C)C)C(=O)O',\n    'P': 'N(CCC1)C1C(=O)O',\n    'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',\n    'I': 'N[C@H]([C@H](C)CC)C(=O)O',\n    'M': 'NC(CCSC)C(=O)O',\n    'T': 'NC(C(O)C)C(=O)O',\n};\nAminoacids.AASmilesTruncated = {\n    'G': '*C*',\n    'L': 'CC(C)C[C@H](*)*',\n    'Y': 'C1=CC(=CC=C1CC(*)*)O',\n    'S': 'OCC(*)C*',\n    'E': '*[C@@H](CCC(O)=O)*',\n    'Q': '*N[C@@H](CCC(N)=O)*',\n    'D': '*[C@@H](CC(O)=O)*',\n    'N': '*[C@@H](CC(N)=O)*',\n    'F': 'C1=CC=C(C=C1)CC(*)*',\n    'A': 'C[C@H](*)*',\n    'K': 'C(CCN)CC(*)*',\n    'R': '*[C@H](CCCNC(=N)C)*',\n    'H': 'C1=C(NC=N1)CC(*)*',\n    'C': 'C([C@@H](*)*)S',\n    'V': 'CC(C)C(*)*',\n    'P': 'C1CCN(*)C1*',\n    'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',\n    'I': 'CC[C@H](C)[C@H](*)*',\n    'M': 'CSCCC(*)*',\n    'T': 'CC(O)C(*)*',\n};\n/** TODO: Full?\n */\nAminoacids.AAFullNames = {\n    'Ala': 'A',\n    'Arg': 'R',\n    'Asn': 'N',\n    'Asp': 'D',\n    'Cys': 'C',\n    'Gln': 'Q',\n    'Glu': 'E',\n    'Gly': 'G',\n    'His': 'H',\n    'Ile': 'I',\n    'Leu': 'L',\n    'Lys': 'K',\n    'Met': 'M',\n    'Phe': 'F',\n    'Pro': 'P',\n    'Ser': 'S',\n    'Thr': 'T',\n    'Trp': 'W',\n    'Tyr': 'Y',\n    'Val': 'V',\n};\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aminoacids.js","sourceRoot":"","sources":["aminoacids.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 {StringDictionary} from '@datagrok-libraries/utils/src/type-declarations';\nimport {SeqPalette, SeqPaletteBase} from './seq-palettes';\n\nexport class AminoacidsPalettes extends SeqPaletteBase {\n  private static lesk: SeqPalette;\n\n  public static get Lesk(): SeqPalette {\n    if (this.lesk === void 0) {\n      this.lesk = this.makePalette([\n        [['G', 'A', 'S', 'T'], 'orange'],\n        [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],\n        [['N', 'Q', 'H'], 'magenta'],\n        [['D', 'E'], 'red'],\n        [['K', 'R'], 'all_blue'],\n      ], false, AminoacidsPalettes);\n    }\n    return this.lesk;\n  }\n\n  private static grokGroups: SeqPalette;\n\n  public static get GrokGroups(): SeqPalette {\n    if (this.grokGroups === void 0) {\n      this.grokGroups = this.makePalette([\n        [['C', 'U'], 'yellow'],\n        [['G', 'P'], 'red'],\n        [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],\n        [['R', 'H', 'K'], 'light_blue'],\n        [['D', 'E'], 'dark_blue'],\n        [['S', 'T', 'N', 'Q'], 'orange'],\n      ], false, AminoacidsPalettes);\n    }\n    return this.grokGroups;\n  }\n\n  private static rasMol: SeqPalette;\n\n  public static get RasMol(): SeqPalette {\n    if (this.rasMol === void 0) {\n      this.rasMol = new AminoacidsPalettes({\n        // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm\n        'D': '#E60A0A', // asp, aspartic acid, asp\n        'E': '#E60A0A', // glu, glutamic acid\n        'C': '#E6E600', // cys, cysteine\n        'M': '#E6E600', // met, methionine\n        'K': '#145AFF', // lys, lysine\n        'R': '#145AFF', // arg, arginine\n        'S': '#FA9600', // ser, serine\n        'T': '#FA9600', // thr, threonine\n        'F': '#3232AA', // phe, phenylalanine\n        'Y': '#3232AA', // tyr, tyrosine\n        'N': '#00DCDC', // asn, asparagine\n        'Q': '#00DCDC', // gln, glutamine\n        'G': '#EBEBEB', // gly, glycine\n        'L': '#0F820F', // leu, leucine\n        'V': '#0F820F', // val, valine\n        'I': '#0F820F', // ile, isoleucine\n        'A': '#C8C8C8', // ala, alanine\n        'W': '#B45AB4', // trp, tryptophan\n        'H': '#8282D2', // his, histidine\n        'P': '#DC9682', // pro, proline\n        'others': '#BEA06E',\n      });\n    }\n    return this.rasMol;\n  }\n}\n\nexport class Aminoacids {\n  static readonly SemType: string = 'Aminoacids';\n\n  static readonly SemTypeMultipleAlignment: string = 'AminoacidsMultipleAlignment';\n\n  static undefinedColor = 'rgb(100,100,100)';\n\n  public static Names: StringDictionary = {\n    'G': 'Glycine',\n    'L': 'Leucine',\n    'Y': 'Tyrosine',\n    'S': 'Serine',\n    'E': 'Glutamic acid',\n    'Q': 'Glutamine',\n    'D': 'Aspartic acid',\n    'N': 'Asparagine',\n    'F': 'Phenylalanine',\n    'A': 'Alanine',\n    'K': 'Lysine',\n    'R': 'Arginine',\n    'H': 'Histidine',\n    'C': 'Cysteine',\n    'V': 'Valine',\n    'P': 'Proline',\n    'W': 'Tryptophan',\n    'I': 'Isoleucine',\n    'M': 'Methionine',\n    'T': 'Threonine',\n  };\n\n  public static AASmiles: StringDictionary = {\n    'G': 'NCC(=O)O',\n    'L': 'N[C@H](CC(C)C)C(=O)O',\n    'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',\n    'S': 'NC(CO)C(=O)O',\n    'E': 'N[C@@H](CCC(O)=O)C(=O)O',\n    'Q': 'N[C@@H](CCC(N)=O)C(=O)O',\n    'D': 'N[C@@H](CC(O)=O)C(=O)O',\n    'N': 'N[C@@H](CC(N)=O)C(=O)O',\n    'F': 'NC(CC1=CC=CC=C1)C(=O)O',\n    'A': 'N[C@H](C)C(=O)O',\n    'K': 'NC(CCCCN)C(=O)O',\n    'R': 'N[C@H](CCCNC(=N)C)C(=O)O',\n    'H': 'NC(CC1=CN=C[N]1)C(=O)O',\n    'C': 'N[C@@H](CS)C(=O)O',\n    'V': 'NC(C(C)C)C(=O)O',\n    'P': 'N(CCC1)C1C(=O)O',\n    'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',\n    'I': 'N[C@H]([C@H](C)CC)C(=O)O',\n    'M': 'NC(CCSC)C(=O)O',\n    'T': 'NC(C(O)C)C(=O)O',\n  };\n\n  public static AASmilesTruncated: StringDictionary = {\n    'G': '*C*',\n    'L': 'CC(C)C[C@H](*)*',\n    'Y': 'C1=CC(=CC=C1CC(*)*)O',\n    'S': 'OCC(*)C*',\n    'E': '*[C@@H](CCC(O)=O)*',\n    'Q': '*N[C@@H](CCC(N)=O)*',\n    'D': '*[C@@H](CC(O)=O)*',\n    'N': '*[C@@H](CC(N)=O)*',\n    'F': 'C1=CC=C(C=C1)CC(*)*',\n    'A': 'C[C@H](*)*',\n    'K': 'C(CCN)CC(*)*',\n    'R': '*[C@H](CCCNC(=N)C)*',\n    'H': 'C1=C(NC=N1)CC(*)*',\n    'C': 'C([C@@H](*)*)S',\n    'V': 'CC(C)C(*)*',\n    'P': 'C1CCN(*)C1*',\n    'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',\n    'I': 'CC[C@H](C)[C@H](*)*',\n    'M': 'CSCCC(*)*',\n    'T': 'CC(O)C(*)*',\n  };\n\n  /** TODO: Full?\n   */\n  public static AAFullNames: StringDictionary = {\n    'Ala': 'A',\n    'Arg': 'R',\n    'Asn': 'N',\n    'Asp': 'D',\n    'Cys': 'C',\n    'Gln': 'Q',\n    'Glu': 'E',\n    'Gly': 'G',\n    'His': 'H',\n    'Ile': 'I',\n    'Leu': 'L',\n    'Lys': 'K',\n    'Met': 'M',\n    'Phe': 'F',\n    'Pro': 'P',\n    'Ser': 'S',\n    'Thr': 'T',\n    'Trp': 'W',\n    'Tyr': 'Y',\n    'Val': 'V',\n  };\n\n  public static getPalette(scheme: string = 'grok'): SeqPalette {\n    switch (scheme) {\n    case 'grok':\n      return AminoacidsPalettes.GrokGroups;\n    case 'lesk':\n      return AminoacidsPalettes.Lesk;\n    default:\n      throw new Error(`ChemPalette: scheme \\`${scheme}\\` does not exist`);\n    }\n  }\n\n  /**\n   * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted\n   *\n   * @param {string} c raw amino\n   * @return {[string, string]} outer and inner content\n   */\n  public static getInnerOuter(c: string): [string, string] {\n    let isInner = 0;\n    let inner = '';\n    let outer = '';\n\n    for (const char of c) {\n      if (char == '(')\n        isInner++;\n      else if (char == ')')\n        isInner--;\n      else if (isInner)\n        inner += char;\n      else\n        outer += char;\n    }\n\n    return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];\n  }\n\n  public static getColorAAPivot(monomer: string = '', scheme: 'grok' = 'grok'): [string, string, string, number] {\n    //const chemPaletteInstance = AAPalettes.GrokGroups();\n    const chemPaletteInstance = this.getPalette(scheme);\n    let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);\n    outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);\n    innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);\n\n    if (monomer.length == 1 || monomer[1] == '(') {\n      const amino = monomer[0]?.toUpperCase()!;\n      return amino in chemPaletteInstance ?\n        [chemPaletteInstance.get(amino), amino, innerMonomer, 1] :\n        [this.undefinedColor, outerMonomer, innerMonomer, 1];\n    }\n\n    if (monomer[0] == 'd' && monomer[1]! in chemPaletteInstance) {\n      if (monomer.length == 2 || monomer[2] == '(') {\n        const amino = monomer[1]?.toUpperCase()!;\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance.get(amino), amino, innerMonomer, 2] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 2];\n      }\n    }\n\n    if (monomer.substring(0, 3) in this.AAFullNames) {\n      if (monomer.length == 3 || monomer[3] == '(') {\n        const amino = this.AAFullNames[monomer.substring(0, 3)];\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance.get(amino), amino, innerMonomer, 3] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 3];\n      }\n    }\n\n    if (monomer[0]?.toLowerCase() == monomer[0]) {\n      if (monomer.substring(1, 3) in this.AAFullNames) {\n        if (monomer.length == 4 || monomer[4] == '(') {\n          const amino = this.AAFullNames[monomer.substring(1, 3)];\n          return amino in chemPaletteInstance ?\n            [chemPaletteInstance.get(amino), amino, innerMonomer, 4] :\n            [this.undefinedColor, outerMonomer, innerMonomer, 4];\n        }\n      }\n    }\n\n    return [this.undefinedColor, outerMonomer, innerMonomer, 0];\n  }\n}\n"]}","import { SeqPaletteBase } from './seq-palettes';\nexport class NucleotidesPalettes extends SeqPaletteBase {\n    static get Chromatogram() {\n        if (this.chromatogram === void 0) {\n            this.chromatogram = new NucleotidesPalettes({\n                'A': 'green',\n                'C': 'blue',\n                'G': 'black',\n                'T': 'red',\n                'U': 'red',\n                'others': 'gray',\n            });\n        }\n        return this.chromatogram;\n    }\n}\nexport class Nucleotides {\n}\nNucleotides.SemType = 'Nucleotides';\nNucleotides.SemTypeMultipleAlignment = 'NucleotidesMultipleAlignment';\nNucleotides.Names = {\n    'A': 'Adenine',\n    'C': 'Cytosine',\n    'G': 'Guanine',\n    'T': 'Thymine',\n    'U': 'Uracil',\n};\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVjbGVvdGlkZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJudWNsZW90aWRlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLQSxPQUFPLEVBQWEsY0FBYyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFMUQsTUFBTSxPQUFPLG1CQUFvQixTQUFRLGNBQWM7SUFHOUMsTUFBTSxLQUFLLFlBQVk7UUFDNUIsSUFBSSxJQUFJLENBQUMsWUFBWSxLQUFLLEtBQUssQ0FBQyxFQUFFO1lBQ2hDLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxtQkFBbUIsQ0FBQztnQkFDMUMsR0FBRyxFQUFFLE9BQU87Z0JBQ1osR0FBRyxFQUFFLE1BQU07Z0JBQ1gsR0FBRyxFQUFFLE9BQU87Z0JBQ1osR0FBRyxFQUFFLEtBQUs7Z0JBQ1YsR0FBRyxFQUFFLEtBQUs7Z0JBQ1YsUUFBUSxFQUFFLE1BQU07YUFDakIsQ0FBQyxDQUFDO1NBQ0o7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUM7SUFDM0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxPQUFPLFdBQVc7O0FBQ04sbUJBQU8sR0FBVyxhQUFhLENBQUM7QUFFaEMsb0NBQXdCLEdBQVcsOEJBQThCLENBQUM7QUFFcEUsaUJBQUssR0FBcUI7SUFDdEMsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsVUFBVTtJQUNmLEdBQUcsRUFBRSxTQUFTO0lBQ2QsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsUUFBUTtDQUNkLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBncm9rIGZyb20gJ2RhdGFncm9rLWFwaS9ncm9rJztcbmltcG9ydCAqIGFzIHVpIGZyb20gJ2RhdGFncm9rLWFwaS91aSc7XG5pbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG5pbXBvcnQge1N0cmluZ0RpY3Rpb25hcnl9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3R5cGUtZGVjbGFyYXRpb25zJztcbmltcG9ydCB7U2VxUGFsZXR0ZSwgU2VxUGFsZXR0ZUJhc2V9IGZyb20gJy4vc2VxLXBhbGV0dGVzJztcblxuZXhwb3J0IGNsYXNzIE51Y2xlb3RpZGVzUGFsZXR0ZXMgZXh0ZW5kcyBTZXFQYWxldHRlQmFzZSB7XG4gIHByaXZhdGUgc3RhdGljIGNocm9tYXRvZ3JhbTogU2VxUGFsZXR0ZTtcblxuICBwdWJsaWMgc3RhdGljIGdldCBDaHJvbWF0b2dyYW0oKTogU2VxUGFsZXR0ZSB7XG4gICAgaWYgKHRoaXMuY2hyb21hdG9ncmFtID09PSB2b2lkIDApIHtcbiAgICAgIHRoaXMuY2hyb21hdG9ncmFtID0gbmV3IE51Y2xlb3RpZGVzUGFsZXR0ZXMoe1xuICAgICAgICAnQSc6ICdncmVlbicsXG4gICAgICAgICdDJzogJ2JsdWUnLFxuICAgICAgICAnRyc6ICdibGFjaycsIC8vIG9yYW5nZSA/XG4gICAgICAgICdUJzogJ3JlZCcsXG4gICAgICAgICdVJzogJ3JlZCcsXG4gICAgICAgICdvdGhlcnMnOiAnZ3JheScsXG4gICAgICB9KTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuY2hyb21hdG9ncmFtO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBOdWNsZW90aWRlcyB7XG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlOiBzdHJpbmcgPSAnTnVjbGVvdGlkZXMnO1xuXG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlTXVsdGlwbGVBbGlnbm1lbnQ6IHN0cmluZyA9ICdOdWNsZW90aWRlc011bHRpcGxlQWxpZ25tZW50JztcblxuICBwdWJsaWMgc3RhdGljIE5hbWVzOiBTdHJpbmdEaWN0aW9uYXJ5ID0ge1xuICAgICdBJzogJ0FkZW5pbmUnLFxuICAgICdDJzogJ0N5dG9zaW5lJyxcbiAgICAnRyc6ICdHdWFuaW5lJyxcbiAgICAnVCc6ICdUaHltaW5lJyxcbiAgICAnVSc6ICdVcmFjaWwnLFxuICB9O1xufVxuIl19","export class SeqPaletteBase {\n    constructor(palette) {\n        this._palette = palette;\n    }\n    static makePalette(dt, simplified = false, PaletteType = SeqPaletteBase) {\n        const palette = {};\n        dt.forEach((cp) => {\n            const objList = cp[0];\n            const colour = cp[1];\n            objList.forEach((obj, ind) => {\n                palette[obj] = this.colourPalette[colour][simplified ? 0 : ind];\n            });\n        });\n        return new PaletteType(palette);\n    }\n    get(m) {\n        return this._palette[m];\n    }\n}\nSeqPaletteBase.undefinedColor = 'rgb(100,100,100)';\n/** Palette with shades of primary colors */\nSeqPaletteBase.colourPalette = {\n    'orange': ['rgb(255,187,120)', 'rgb(245,167,100)', 'rgb(235,137,70)', 'rgb(205, 111, 71)'],\n    'all_green': ['rgb(44,160,44)', 'rgb(74,160,74)', 'rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)',\n        'rgb(24,110,79)', 'rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],\n    'all_blue': ['rgb(31,119,180)', 'rgb(23,190,207)', 'rgb(122, 102, 189)', 'rgb(158,218,229)', 'rgb(141, 124, 217)',\n        'rgb(31, 120, 150)'],\n    'magenta': ['rgb(162,106,192)', 'rgb(197,165,224)', 'rgb(208,113,218)'],\n    'red': ['rgb(214,39,40)', 'rgb(255,152,150)'],\n    'st_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(31,119,180)'],\n    'dark_blue': ['rgb(31,119,180)', 'rgb(31, 120, 150)'],\n    'light_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(108, 218, 229)', 'rgb(23,190,227)'],\n    'lilac_blue': ['rgb(124,102,211)', 'rgb(149,134,217)', 'rgb(97, 81, 150)'],\n    'dark_green': ['rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)', 'rgb(24,110,79)'],\n    'green': ['rgb(44,160,44)', 'rgb(74,160,74)'],\n    'light_green': ['rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],\n    'st_green': ['rgb(44,160,44)', 'rgb(152,223,138)', 'rgb(39, 174, 96)', 'rgb(74,160,74)'],\n    'pink': ['rgb(247,182,210)'],\n    'brown': ['rgb(140,86,75)', 'rgb(102, 62, 54)'],\n    'gray': ['rgb(127,127,127)', 'rgb(199,199,199)', 'rgb(196,156,148)', 'rgb(222, 222, 180)'],\n    'yellow': ['rgb(188,189,34)'],\n    'white': ['rgb(230,230,230)'],\n};\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxLXBhbGV0dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic2VxLXBhbGV0dGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWlCQSxNQUFNLE9BQU8sY0FBYztJQTJDekIsWUFBWSxPQUFnQztRQUMxQyxJQUFJLENBQUMsUUFBUSxHQUFHLE9BQU8sQ0FBQztJQUMxQixDQUFDO0lBbEJTLE1BQU0sQ0FBQyxXQUFXLENBQUMsRUFBd0IsRUFDbkQsVUFBVSxHQUFHLEtBQUssRUFBRSxjQUFxQyxjQUFjO1FBRXZFLE1BQU0sT0FBTyxHQUE4QixFQUFFLENBQUM7UUFDOUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxDQUFDLEVBQUUsRUFBRSxFQUFFO1lBQ2hCLE1BQU0sT0FBTyxHQUFHLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN0QixNQUFNLE1BQU0sR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDckIsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsRUFBRTtnQkFDM0IsT0FBTyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ2xFLENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7UUFDSCxPQUFPLElBQUksV0FBVyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFRTSxHQUFHLENBQUMsQ0FBUztRQUNsQixPQUFPLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDMUIsQ0FBQzs7QUFoRGEsNkJBQWMsR0FBRyxrQkFBa0IsQ0FBQztBQUVsRCw0Q0FBNEM7QUFDOUIsNEJBQWEsR0FBZ0M7SUFDekQsUUFBUSxFQUFFLENBQUMsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsaUJBQWlCLEVBQUUsbUJBQW1CLENBQUM7SUFDMUYsV0FBVyxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCO1FBQ3BHLGdCQUFnQixFQUFFLGtCQUFrQixFQUFFLG9CQUFvQixFQUFFLG9CQUFvQixDQUFDO0lBQ25GLFVBQVUsRUFBRSxDQUFDLGlCQUFpQixFQUFFLGlCQUFpQixFQUFFLG9CQUFvQixFQUFFLGtCQUFrQixFQUFFLG9CQUFvQjtRQUMvRyxtQkFBbUIsQ0FBQztJQUN0QixTQUFTLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsQ0FBQztJQUN2RSxLQUFLLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsQ0FBQztJQUM3QyxTQUFTLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxrQkFBa0IsRUFBRSxpQkFBaUIsQ0FBQztJQUNyRSxXQUFXLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxtQkFBbUIsQ0FBQztJQUNyRCxZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxpQkFBaUIsQ0FBQztJQUM5RixZQUFZLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsQ0FBQztJQUMxRSxZQUFZLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxnQkFBZ0IsRUFBRSxnQkFBZ0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUN0RixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUM3QyxhQUFhLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxvQkFBb0IsQ0FBQztJQUMvRSxVQUFVLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUN4RixNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztJQUM1QixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsQ0FBQztJQUMvQyxNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsQ0FBQztJQUMxRixRQUFRLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztJQUM3QixPQUFPLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztDQUM5QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgZ3JvayBmcm9tICdkYXRhZ3Jvay1hcGkvZ3Jvayc7XG5pbXBvcnQgKiBhcyB1aSBmcm9tICdkYXRhZ3Jvay1hcGkvdWknO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuZXhwb3J0IGludGVyZmFjZSBTZXFQYWxldHRlIHtcbiAgLy8gVGhlcmUgYXJlIHRvbyBtdWNoIHByb2JsZW0gd2l0aCBpbmRleGVyIHByb3BlcnR5IGluIHR5cGVzY3JpcHQuXG4gIC8vIC8qKlxuICAvLyAgKiBAcGFyYW0ge3N0cmluZ30gbSBNb25vbWVyIGNoYXJhY3RlclxuICAvLyAgKiBAcmV0dXJuIHtzdHJpbmd9IENvbG9yXG4gIC8vICAqL1xuICAvLyBbbTogc3RyaW5nXTogc3RyaW5nO1xuICAvKiogTW9ub21lciBjb2xvclxuICAgKiBAcGFyYW0ge3N0cmluZ30gbSBNb25vbWVyXG4gICAqL1xuICBnZXQobTogc3RyaW5nKTogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgU2VxUGFsZXR0ZUJhc2UgaW1wbGVtZW50cyBTZXFQYWxldHRlIHtcbiAgcHVibGljIHN0YXRpYyB1bmRlZmluZWRDb2xvciA9ICdyZ2IoMTAwLDEwMCwxMDApJztcblxuICAvKiogUGFsZXR0ZSB3aXRoIHNoYWRlcyBvZiBwcmltYXJ5IGNvbG9ycyAqL1xuICBwdWJsaWMgc3RhdGljIGNvbG91clBhbGV0dGU6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nW10gfSA9IHtcbiAgICAnb3JhbmdlJzogWydyZ2IoMjU1LDE4NywxMjApJywgJ3JnYigyNDUsMTY3LDEwMCknLCAncmdiKDIzNSwxMzcsNzApJywgJ3JnYigyMDUsIDExMSwgNzEpJ10sXG4gICAgJ2FsbF9ncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDc0LDE2MCw3NCknLCAncmdiKDIzLDEwMyw1NyknLCAncmdiKDMwLDExMCw5NiknLCAncmdiKDYwLDEzMSw5NSknLFxuICAgICAgJ3JnYigyNCwxMTAsNzkpJywgJ3JnYigxNTIsMjIzLDEzOCknLCAncmdiKDE4MiwgMjIzLCAxMzgpJywgJ3JnYigxNTIsIDE5MywgMTM4KSddLFxuICAgICdhbGxfYmx1ZSc6IFsncmdiKDMxLDExOSwxODApJywgJ3JnYigyMywxOTAsMjA3KScsICdyZ2IoMTIyLCAxMDIsIDE4OSknLCAncmdiKDE1OCwyMTgsMjI5KScsICdyZ2IoMTQxLCAxMjQsIDIxNyknLFxuICAgICAgJ3JnYigzMSwgMTIwLCAxNTApJ10sXG4gICAgJ21hZ2VudGEnOiBbJ3JnYigxNjIsMTA2LDE5MiknLCAncmdiKDE5NywxNjUsMjI0KScsICdyZ2IoMjA4LDExMywyMTgpJ10sXG4gICAgJ3JlZCc6IFsncmdiKDIxNCwzOSw0MCknLCAncmdiKDI1NSwxNTIsMTUwKSddLFxuICAgICdzdF9ibHVlJzogWydyZ2IoMjMsMTkwLDIwNyknLCAncmdiKDE1OCwyMTgsMjI5KScsICdyZ2IoMzEsMTE5LDE4MCknXSxcbiAgICAnZGFya19ibHVlJzogWydyZ2IoMzEsMTE5LDE4MCknLCAncmdiKDMxLCAxMjAsIDE1MCknXSxcbiAgICAnbGlnaHRfYmx1ZSc6IFsncmdiKDIzLDE5MCwyMDcpJywgJ3JnYigxNTgsMjE4LDIyOSknLCAncmdiKDEwOCwgMjE4LCAyMjkpJywgJ3JnYigyMywxOTAsMjI3KSddLFxuICAgICdsaWxhY19ibHVlJzogWydyZ2IoMTI0LDEwMiwyMTEpJywgJ3JnYigxNDksMTM0LDIxNyknLCAncmdiKDk3LCA4MSwgMTUwKSddLFxuICAgICdkYXJrX2dyZWVuJzogWydyZ2IoMjMsMTAzLDU3KScsICdyZ2IoMzAsMTEwLDk2KScsICdyZ2IoNjAsMTMxLDk1KScsICdyZ2IoMjQsMTEwLDc5KSddLFxuICAgICdncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDc0LDE2MCw3NCknXSxcbiAgICAnbGlnaHRfZ3JlZW4nOiBbJ3JnYigxNTIsMjIzLDEzOCknLCAncmdiKDE4MiwgMjIzLCAxMzgpJywgJ3JnYigxNTIsIDE5MywgMTM4KSddLFxuICAgICdzdF9ncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDE1MiwyMjMsMTM4KScsICdyZ2IoMzksIDE3NCwgOTYpJywgJ3JnYig3NCwxNjAsNzQpJ10sXG4gICAgJ3BpbmsnOiBbJ3JnYigyNDcsMTgyLDIxMCknXSxcbiAgICAnYnJvd24nOiBbJ3JnYigxNDAsODYsNzUpJywgJ3JnYigxMDIsIDYyLCA1NCknXSxcbiAgICAnZ3JheSc6IFsncmdiKDEyNywxMjcsMTI3KScsICdyZ2IoMTk5LDE5OSwxOTkpJywgJ3JnYigxOTYsMTU2LDE0OCknLCAncmdiKDIyMiwgMjIyLCAxODApJ10sXG4gICAgJ3llbGxvdyc6IFsncmdiKDE4OCwxODksMzQpJ10sXG4gICAgJ3doaXRlJzogWydyZ2IoMjMwLDIzMCwyMzApJ10sXG4gIH07XG5cbiAgcHJvdGVjdGVkIHN0YXRpYyBtYWtlUGFsZXR0ZShkdDogW3N0cmluZ1tdLCBzdHJpbmddW10sXG4gICAgc2ltcGxpZmllZCA9IGZhbHNlLCBQYWxldHRlVHlwZTogdHlwZW9mIFNlcVBhbGV0dGVCYXNlID0gU2VxUGFsZXR0ZUJhc2VcbiAgKSB7XG4gICAgY29uc3QgcGFsZXR0ZTogeyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfSA9IHt9O1xuICAgIGR0LmZvckVhY2goKGNwKSA9PiB7XG4gICAgICBjb25zdCBvYmpMaXN0ID0gY3BbMF07XG4gICAgICBjb25zdCBjb2xvdXIgPSBjcFsxXTtcbiAgICAgIG9iakxpc3QuZm9yRWFjaCgob2JqLCBpbmQpID0+IHtcbiAgICAgICAgcGFsZXR0ZVtvYmpdID0gdGhpcy5jb2xvdXJQYWxldHRlW2NvbG91cl1bc2ltcGxpZmllZCA/IDAgOiBpbmRdO1xuICAgICAgfSk7XG4gICAgfSk7XG4gICAgcmV0dXJuIG5ldyBQYWxldHRlVHlwZShwYWxldHRlKTtcbiAgfVxuXG4gIHByaXZhdGUgcmVhZG9ubHkgX3BhbGV0dGU6IHsgW206IHN0cmluZ106IHN0cmluZyB9O1xuXG4gIGNvbnN0cnVjdG9yKHBhbGV0dGU6IHsgW206IHN0cmluZ106IHN0cmluZyB9KSB7XG4gICAgdGhpcy5fcGFsZXR0ZSA9IHBhbGV0dGU7XG4gIH1cblxuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuX3BhbGV0dGVbbV07XG4gIH1cbn1cbiJdfQ==","import { assert } from '@datagrok-libraries/utils/src/vector-operations';\nclass SideChainScales {\n    static getAvailableScales() {\n        return Object.entries(this.scales).map(([k, _]) => k);\n    }\n    static getScale(name) {\n        assert(!(this.scales[name] === undefined), `Scale '${name}' was not found.`);\n        return this.scales[name];\n    }\n}\nSideChainScales.scales = {\n    // Wimley-White interfacial hydrophobicity scale\n    'WimleyWhite': {\n        '-': 0,\n        'A': 0.17,\n        'C': -0.24,\n        'D': -0.07,\n        'E': -0.01,\n        'F': -1.13,\n        'G': 0.01,\n        'H': 0.17,\n        'I': -0.31,\n        'K': 0.99,\n        'L': -0.56,\n        'M': -0.23,\n        'N': 0.42,\n        'P': 0.45,\n        'Q': 0.58,\n        'R': 0.81,\n        'S': 0.13,\n        'T': 0.14,\n        'V': 0.07,\n        'W': -1.85,\n        'Y': -0.94,\n    },\n    'categorial': {\n        '-': 0,\n        'A': 1,\n        'C': 2,\n        'D': 3,\n        'E': 4,\n        'F': 5,\n        'G': 6,\n        'H': 7,\n        'I': 8,\n        'K': 9,\n        'L': 10,\n        'M': 11,\n        'N': 12,\n        'P': 13,\n        'Q': 14,\n        'R': 15,\n        'S': 16,\n        'T': 17,\n        'V': 18,\n        'W': 19,\n        'Y': 20,\n    },\n};\n/**\n * Class to categorial encode/decode aligned amino acid residues sequence.\n *\n * @export\n * @class AlignedSequenceEncoder\n */\nexport class AlignedSequenceEncoder {\n    constructor(scale = 'categorial') {\n        this.aa2num = SideChainScales.getScale(scale);\n        this.num2aa = {};\n        Object.entries(this.aa2num).forEach(([k, v]) => (this.num2aa[v] = k));\n    }\n    /**\n       * Truncate NH2 and -COOH terminals of the given sequence.\n       *\n       * @static\n       * @param {string} seq The sequence provided.\n       * @return {string} Truncated sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    static _truncateSequence(seq) {\n        let start = 0;\n        let end = seq.length;\n        const termina = ['NH2', 'COOH'];\n        if (seq.startsWith(termina[0])) {\n            const l = termina[0].length; // Cut only 'NH2' without following '-'.\n            assert(seq[l] == '-', `Wrong sequence format: ${termina[0]} without following '-' in '${seq}'.`);\n            start = l;\n        }\n        if (seq.endsWith(termina[1])) {\n            const l = termina[1].length + 1; // Cut both 'COOH' and precending '-'.\n            assert(seq[end - l] == '-', `Wrong sequence format: ${termina[1]} without '-' precending in '${seq}'.`);\n            end -= l;\n        }\n        return seq.substring(start, end);\n    }\n    /**\n       * Cuts auxiliary defises before a residue.\n       *\n       * @static\n       * @param {string} seq The sequence to process.\n       * @return {string} Processed sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    static _dropDefises(seq) {\n        return seq.replace(/(-)([^-]+)/g, '$2');\n    }\n    /**\n       * Performs truncation and cutting auxiliary defises.\n       *\n       * @static\n       * @param {string} sequence The sequence work under process.\n       * @return {string} Result of cleaning.\n       * @memberof AlignedSequenceEncoder\n       */\n    static clean(sequence) {\n        return AlignedSequenceEncoder._dropDefises(AlignedSequenceEncoder._truncateSequence(sequence));\n    }\n    /**\n       * Categorial encode of the sequence provided.\n       *\n       * @param {string} sequence The sequence.\n       * @return {number[]} Encoded vector.\n       * @memberof AlignedSequenceEncoder\n       */\n    encode(sequence) {\n        const nItems = sequence.length;\n        const values = new Array(nItems).fill(0);\n        for (let i = 0; i < nItems; ++i) {\n            const char = sequence[i];\n            assert(char in this.aa2num, `Unknown char '${char}' found in sequence '${sequence}'`);\n            values[i] = this.encodeLettter(char);\n        }\n        return values;\n    }\n    encodeLettter(letter) {\n        return this.aa2num[letter];\n    }\n    /**\n       * Decode the encoded vector into the sequence back.\n       *\n       * @param {number[]} value The vector encoded.\n       * @return {string} Decoded sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    decode(value) {\n        let s = '';\n        for (let i = 0; i < value.length; ++i) {\n            const code = value[i];\n            assert(code in this.num2aa, `Unknown code '${code}' found in vector '${value}'`);\n            s += this.num2aa[code];\n        }\n        return s;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sequence-encoder.js","sourceRoot":"","sources":["sequence-encoder.ts"],"names":[],"mappings":"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","sourcesContent":["import {assert} from '@datagrok-libraries/utils/src/vector-operations';\n\ntype SideChainScale = {[name: string]: number};\ntype SideChainScaleCollection = {[name: string]: SideChainScale};\n\nclass SideChainScales {\n  static scales: SideChainScaleCollection = {\n    // Wimley-White interfacial hydrophobicity scale\n    'WimleyWhite': {\n      '-': 0,\n      'A': 0.17,\n      'C': -0.24,\n      'D': -0.07, // Asp-: 1.23\n      'E': -0.01, // Glu-: 2.02\n      'F': -1.13, //\n      'G': 0.01,\n      'H': 0.17, // His+: 0.96\n      'I': -0.31,\n      'K': 0.99, // Lys+\n      'L': -0.56,\n      'M': -0.23,\n      'N': 0.42,\n      'P': 0.45,\n      'Q': 0.58,\n      'R': 0.81, // Arg+\n      'S': 0.13,\n      'T': 0.14,\n      'V': 0.07,\n      'W': -1.85,\n      'Y': -0.94,\n    },\n    'categorial': {\n      '-': 0,\n      'A': 1,\n      'C': 2,\n      'D': 3,\n      'E': 4,\n      'F': 5,\n      'G': 6,\n      'H': 7,\n      'I': 8,\n      'K': 9,\n      'L': 10,\n      'M': 11,\n      'N': 12,\n      'P': 13,\n      'Q': 14,\n      'R': 15,\n      'S': 16,\n      'T': 17,\n      'V': 18,\n      'W': 19,\n      'Y': 20,\n    },\n  };\n\n  static getAvailableScales(): string[] {\n    return Object.entries(this.scales).map(([k, _]) => k);\n  }\n\n  static getScale(name: string): SideChainScale {\n    assert(!(this.scales[name] === undefined), `Scale '${name}' was not found.`);\n    return this.scales[name];\n  }\n}\n\n/**\n * Class to categorial encode/decode aligned amino acid residues sequence.\n *\n * @export\n * @class AlignedSequenceEncoder\n */\nexport class AlignedSequenceEncoder {\n  protected aa2num: SideChainScale;\n  protected num2aa: {[code: number]: string};\n\n  constructor(scale: string = 'categorial') {\n    this.aa2num = SideChainScales.getScale(scale);\n    this.num2aa = {};\n    Object.entries(this.aa2num).forEach(([k, v]) => (this.num2aa[v] = k));\n  }\n\n  /**\n     * Truncate NH2 and -COOH terminals of the given sequence.\n     *\n     * @static\n     * @param {string} seq The sequence provided.\n     * @return {string} Truncated sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  static _truncateSequence(seq: string): string {\n    let start = 0;\n    let end = seq.length;\n    const termina = ['NH2', 'COOH'];\n\n    if (seq.startsWith(termina[0])) {\n      const l = termina[0].length; // Cut only 'NH2' without following '-'.\n      assert(seq[l] == '-', `Wrong sequence format: ${termina[0]} without following '-' in '${seq}'.`);\n      start = l;\n    }\n    if (seq.endsWith(termina[1])) {\n      const l = termina[1].length+1; // Cut both 'COOH' and precending '-'.\n      assert(seq[end-l] == '-', `Wrong sequence format: ${termina[1]} without '-' precending in '${seq}'.`);\n      end -= l;\n    }\n    return seq.substring(start, end);\n  }\n\n  /**\n     * Cuts auxiliary defises before a residue.\n     *\n     * @static\n     * @param {string} seq The sequence to process.\n     * @return {string} Processed sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  static _dropDefises(seq: string): string {\n    return seq.replace(/(-)([^-]+)/g, '$2');\n  }\n\n  /**\n     * Performs truncation and cutting auxiliary defises.\n     *\n     * @static\n     * @param {string} sequence The sequence work under process.\n     * @return {string} Result of cleaning.\n     * @memberof AlignedSequenceEncoder\n     */\n  static clean(sequence: string): string {\n    return AlignedSequenceEncoder._dropDefises(AlignedSequenceEncoder._truncateSequence(sequence));\n  }\n\n  /**\n     * Categorial encode of the sequence provided.\n     *\n     * @param {string} sequence The sequence.\n     * @return {number[]} Encoded vector.\n     * @memberof AlignedSequenceEncoder\n     */\n  public encode(sequence: string): number[] {\n    const nItems = sequence.length;\n    const values = new Array(nItems).fill(0);\n\n    for (let i = 0; i < nItems; ++i) {\n      const char = sequence[i];\n\n      assert(char in this.aa2num, `Unknown char '${char}' found in sequence '${sequence}'`);\n\n      values[i] = this.encodeLettter(char);\n    }\n    return values;\n  }\n\n  public encodeLettter(letter: string): number {\n    return this.aa2num[letter];\n  }\n\n  /**\n     * Decode the encoded vector into the sequence back.\n     *\n     * @param {number[]} value The vector encoded.\n     * @return {string} Decoded sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  public decode(value: number[]): string {\n    let s: string = '';\n\n    for (let i = 0; i < value.length; ++i) {\n      const code = value[i];\n\n      assert(code in this.num2aa, `Unknown code '${code}' found in vector '${value}'`);\n\n      s += this.num2aa[code];\n    }\n    return s;\n  }\n}\n"]}","export class StringUtils {\n    static hashCode(s) {\n        let hash = 0;\n        if (s.length === 0)\n            return hash;\n        for (let i = 0; i < s.length; i++) {\n            const chr = s.charCodeAt(i);\n            hash = ((hash << 5) - hash) + chr;\n            hash |= 0; // Convert to 32bit integer\n        }\n        return hash;\n    }\n}\nimport { SeqPaletteBase } from './seq-palettes';\nexport class UnknownSeqPalette {\n}\nexport class GrayAllPalette extends UnknownSeqPalette {\n    get(m) {\n        return '#666666';\n    }\n}\nexport class UnknownColorPalette extends UnknownSeqPalette {\n    static buildPalette() {\n        const res = [].concat(...Object.values(SeqPaletteBase.colourPalette));\n        return res;\n    }\n    get(m) {\n        const hash = StringUtils.hashCode(m);\n        const pI = hash % UnknownColorPalette.palette.length;\n        return UnknownColorPalette.palette[pI];\n    }\n}\nUnknownColorPalette.palette = UnknownColorPalette.buildPalette();\nexport class UnknownSeqPalettes extends SeqPaletteBase {\n    static get Gray() {\n        if (this.gray === void 0)\n            this.gray = new GrayAllPalette();\n        return this.gray;\n    }\n    static get Color() {\n        if (this.color === void 0)\n            this.color = new UnknownColorPalette();\n        return this.color;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5rbm93bi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInVua25vd24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBSUEsTUFBTSxPQUFPLFdBQVc7SUFDZixNQUFNLENBQUMsUUFBUSxDQUFDLENBQVM7UUFDOUIsSUFBSSxJQUFJLEdBQVcsQ0FBQyxDQUFDO1FBQ3JCLElBQUksQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDO1lBQ2hCLE9BQU8sSUFBSSxDQUFDO1FBQ2QsS0FBSyxJQUFJLENBQUMsR0FBVyxDQUFDLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUU7WUFDekMsTUFBTSxHQUFHLEdBQVcsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNwQyxJQUFJLEdBQUcsQ0FBQyxDQUFDLElBQUksSUFBSSxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxHQUFHLENBQUM7WUFDbEMsSUFBSSxJQUFJLENBQUMsQ0FBQyxDQUFDLDJCQUEyQjtTQUN2QztRQUNELE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztDQUNGO0FBRUQsT0FBTyxFQUFhLGNBQWMsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRTFELE1BQU0sT0FBZ0IsaUJBQWlCO0NBRXRDO0FBRUQsTUFBTSxPQUFPLGNBQWUsU0FBUSxpQkFBaUI7SUFDNUMsR0FBRyxDQUFDLENBQVM7UUFDbEIsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztDQUNGO0FBR0QsTUFBTSxPQUFPLG1CQUFvQixTQUFRLGlCQUFpQjtJQUdoRCxNQUFNLENBQUMsWUFBWTtRQUN6QixNQUFNLEdBQUcsR0FBSSxFQUFlLENBQUMsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQztRQUNwRixPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7SUFFTSxHQUFHLENBQUMsQ0FBUztRQUNsQixNQUFNLElBQUksR0FBVyxXQUFXLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzdDLE1BQU0sRUFBRSxHQUFHLElBQUksR0FBRyxtQkFBbUIsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDO1FBQ3JELE9BQU8sbUJBQW1CLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3pDLENBQUM7O0FBWGEsMkJBQU8sR0FBYSxtQkFBbUIsQ0FBQyxZQUFZLEVBQUUsQ0FBQztBQWN2RSxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsY0FBYztJQUc3QyxNQUFNLEtBQUssSUFBSTtRQUNwQixJQUFJLElBQUksQ0FBQyxJQUFJLEtBQUssS0FBSyxDQUFDO1lBQ3RCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxjQUFjLEVBQUUsQ0FBQztRQUNuQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDbkIsQ0FBQztJQUlNLE1BQU0sS0FBSyxLQUFLO1FBQ3JCLElBQUksSUFBSSxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUM7WUFDdkIsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLG1CQUFtQixFQUFFLENBQUM7UUFDekMsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgdWkgZnJvbSAnZGF0YWdyb2stYXBpL3VpJztcbmltcG9ydCAqIGFzIERHIGZyb20gJ2RhdGFncm9rLWFwaS9kZyc7XG5cbmV4cG9ydCBjbGFzcyBTdHJpbmdVdGlscyB7XG4gIHB1YmxpYyBzdGF0aWMgaGFzaENvZGUoczogc3RyaW5nKTogbnVtYmVyIHtcbiAgICBsZXQgaGFzaDogbnVtYmVyID0gMDtcbiAgICBpZiAocy5sZW5ndGggPT09IDApXG4gICAgICByZXR1cm4gaGFzaDtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcy5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgY2hyOiBudW1iZXIgPSBzLmNoYXJDb2RlQXQoaSk7XG4gICAgICBoYXNoID0gKChoYXNoIDw8IDUpIC0gaGFzaCkgKyBjaHI7XG4gICAgICBoYXNoIHw9IDA7IC8vIENvbnZlcnQgdG8gMzJiaXQgaW50ZWdlclxuICAgIH1cbiAgICByZXR1cm4gaGFzaDtcbiAgfVxufVxuXG5pbXBvcnQge1NlcVBhbGV0dGUsIFNlcVBhbGV0dGVCYXNlfSBmcm9tICcuL3NlcS1wYWxldHRlcyc7XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBVbmtub3duU2VxUGFsZXR0ZSBpbXBsZW1lbnRzIFNlcVBhbGV0dGUge1xuICBwdWJsaWMgYWJzdHJhY3QgZ2V0KG06IHN0cmluZyk6IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIEdyYXlBbGxQYWxldHRlIGV4dGVuZHMgVW5rbm93blNlcVBhbGV0dGUge1xuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuICcjNjY2NjY2JztcbiAgfVxufVxuXG5cbmV4cG9ydCBjbGFzcyBVbmtub3duQ29sb3JQYWxldHRlIGV4dGVuZHMgVW5rbm93blNlcVBhbGV0dGUge1xuICBwdWJsaWMgc3RhdGljIHBhbGV0dGU6IHN0cmluZ1tdID0gVW5rbm93bkNvbG9yUGFsZXR0ZS5idWlsZFBhbGV0dGUoKTtcblxuICBwcml2YXRlIHN0YXRpYyBidWlsZFBhbGV0dGUoKTogc3RyaW5nW10ge1xuICAgIGNvbnN0IHJlcyA9IChbXSBhcyBzdHJpbmdbXSkuY29uY2F0KC4uLk9iamVjdC52YWx1ZXMoU2VxUGFsZXR0ZUJhc2UuY29sb3VyUGFsZXR0ZSkpO1xuICAgIHJldHVybiByZXM7XG4gIH1cblxuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3QgaGFzaDogbnVtYmVyID0gU3RyaW5nVXRpbHMuaGFzaENvZGUobSk7XG4gICAgY29uc3QgcEkgPSBoYXNoICUgVW5rbm93bkNvbG9yUGFsZXR0ZS5wYWxldHRlLmxlbmd0aDtcbiAgICByZXR1cm4gVW5rbm93bkNvbG9yUGFsZXR0ZS5wYWxldHRlW3BJXTtcbiAgfVxufVxuXG5leHBvcnQgY2xhc3MgVW5rbm93blNlcVBhbGV0dGVzIGV4dGVuZHMgU2VxUGFsZXR0ZUJhc2Uge1xuICBwcml2YXRlIHN0YXRpYyBncmF5OiBTZXFQYWxldHRlO1xuXG4gIHB1YmxpYyBzdGF0aWMgZ2V0IEdyYXkoKTogU2VxUGFsZXR0ZSB7XG4gICAgaWYgKHRoaXMuZ3JheSA9PT0gdm9pZCAwKVxuICAgICAgdGhpcy5ncmF5ID0gbmV3IEdyYXlBbGxQYWxldHRlKCk7XG4gICAgcmV0dXJuIHRoaXMuZ3JheTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNvbG9yOiBTZXFQYWxldHRlO1xuXG4gIHB1YmxpYyBzdGF0aWMgZ2V0IENvbG9yKCk6IFNlcVBhbGV0dGUge1xuICAgIGlmICh0aGlzLmNvbG9yID09PSB2b2lkIDApXG4gICAgICB0aGlzLmNvbG9yID0gbmV3IFVua25vd25Db2xvclBhbGV0dGUoKTtcbiAgICByZXR1cm4gdGhpcy5jb2xvcjtcbiAgfVxufVxuIl19","const undefinedColor = 'rgb(100,100,100)';\nconst grayColor = '#808080';\nexport var DrawStyle;\n(function (DrawStyle) {\n    DrawStyle[\"MSA\"] = \"MSA\";\n    DrawStyle[\"classic\"] = \"classic\";\n})(DrawStyle || (DrawStyle = {}));\n/**\n * A function that prints a string aligned to left or centered.\n *\n * @param {number} x x coordinate.\n * @param {number} y y coordinate.\n * @param {number} w Width.\n * @param {number} h Height.\n * @param {CanvasRenderingContext2D} g Canvas rendering context.\n * @param {string} s String to print.\n * @param {string} [color=undefinedColor] String color.\n * @param {number} [pivot=0] Pirvot.\n * @param {boolean} [left=false] Is left aligned.\n * @param {number} [transparencyRate=0.0] Transparency rate where 1.0 is fully transparent\n * @param {string} [separator=''] Is separator for sequence.\n * @param {boolean} [last=false] Is checker if element last or not.\n * @param drawStyle Is draw style. MSA - for multicharSeq, classic - for other seq.\n * @param maxWord Is array of max words for each line.\n * @param maxWordIdx Is index of word we currently draw.\n * @param gridCell Is grid cell, new for updating data in maxWord while rendering.\n * @return {number} x coordinate to start printing at.\n */\nexport function printLeftOrCentered(x, y, w, h, g, s, color = undefinedColor, pivot = 0, left = false, transparencyRate = 1.0, separator = '', last = false, drawStyle = DrawStyle.classic, maxWord = {}, maxWordIdx = 0, gridCell = {}) {\n    var _a;\n    g.textAlign = 'start';\n    const colorPart = s.substring(0);\n    let grayPart = last ? '' : separator;\n    if (drawStyle === DrawStyle.MSA) {\n        grayPart = '';\n    }\n    let textSize = g.measureText(colorPart + grayPart);\n    const indent = 5;\n    let maxColorTextSize = g.measureText(colorPart).width;\n    let colorTextSize = g.measureText(colorPart).width;\n    const dy = (textSize.fontBoundingBoxAscent + textSize.fontBoundingBoxDescent) / 2;\n    textSize = textSize.width;\n    if (drawStyle === DrawStyle.MSA) {\n        maxColorTextSize = maxWord[maxWordIdx];\n        textSize = maxWord[maxWordIdx];\n        if (maxWordIdx > ((_a = maxWord['bio-maxIndex']) !== null && _a !== void 0 ? _a : 0)) {\n            maxWord['bio-maxIndex'] = maxWordIdx;\n            gridCell.cell.column.temp = maxWord;\n        }\n    }\n    function draw(dx1, dx2) {\n        g.fillStyle = color;\n        g.globalAlpha = transparencyRate;\n        if (drawStyle === DrawStyle.classic) {\n            g.fillText(colorPart, x + dx1, y + dy);\n            g.fillStyle = grayColor;\n            g.fillText(grayPart, x + dx2, y + dy);\n        }\n        if (drawStyle === DrawStyle.MSA) {\n            g.fillStyle = color;\n            g.fillText(colorPart, x + dx1 + ((maxWord[maxWordIdx] - colorTextSize) / 2), y + dy);\n        }\n    }\n    if (left || textSize > w) {\n        draw(indent, indent + maxColorTextSize);\n        return x + maxColorTextSize + g.measureText(grayPart).width;\n    }\n    else {\n        const dx = (w - textSize) / 2;\n        draw(dx, dx + maxColorTextSize);\n        return x + dx + maxColorTextSize;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2VsbC1yZW5kZXJlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImNlbGwtcmVuZGVyZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxjQUFjLEdBQUcsa0JBQWtCLENBQUM7QUFDMUMsTUFBTSxTQUFTLEdBQUcsU0FBUyxDQUFDO0FBRTVCLE1BQU0sQ0FBTixJQUFZLFNBR1g7QUFIRCxXQUFZLFNBQVM7SUFDbkIsd0JBQVcsQ0FBQTtJQUNYLGdDQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFIVyxTQUFTLEtBQVQsU0FBUyxRQUdwQjtBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9CRztBQUNILE1BQU0sVUFBVSxtQkFBbUIsQ0FDakMsQ0FBUyxFQUFFLENBQVMsRUFBRSxDQUFTLEVBQUUsQ0FBUyxFQUMxQyxDQUEyQixFQUFFLENBQVMsRUFBRSxLQUFLLEdBQUcsY0FBYyxFQUM5RCxRQUFnQixDQUFDLEVBQUUsSUFBSSxHQUFHLEtBQUssRUFBRSxtQkFBMkIsR0FBRyxFQUMvRCxZQUFvQixFQUFFLEVBQUUsT0FBZ0IsS0FBSyxFQUFFLFlBQXVCLFNBQVMsQ0FBQyxPQUFPLEVBQUUsVUFBdUMsRUFBRSxFQUFFLGFBQXFCLENBQUMsRUFBRSxXQUFnQixFQUFFOztJQUM5SyxDQUFDLENBQUMsU0FBUyxHQUFHLE9BQU8sQ0FBQztJQUN0QixNQUFNLFNBQVMsR0FBRyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2pDLElBQUksUUFBUSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7SUFDckMsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLEdBQUcsRUFBRTtRQUMvQixRQUFRLEdBQUcsRUFBRSxDQUFDO0tBQ2Y7SUFFRCxJQUFJLFFBQVEsR0FBUSxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsR0FBRyxRQUFRLENBQUMsQ0FBQztJQUN4RCxNQUFNLE1BQU0sR0FBRyxDQUFDLENBQUM7SUFFakIsSUFBSSxnQkFBZ0IsR0FBRyxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEtBQUssQ0FBQztJQUN0RCxJQUFJLGFBQWEsR0FBRyxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEtBQUssQ0FBQztJQUNuRCxNQUFNLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxxQkFBcUIsR0FBRyxRQUFRLENBQUMsc0JBQXNCLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDbEYsUUFBUSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUM7SUFDMUIsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLEdBQUcsRUFBRTtRQUMvQixnQkFBZ0IsR0FBRyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDdkMsUUFBUSxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUMvQixJQUFJLFVBQVUsR0FBRyxDQUFDLE1BQUEsT0FBTyxDQUFDLGNBQWMsQ0FBQyxtQ0FBSSxDQUFDLENBQUMsRUFBRTtZQUMvQyxPQUFPLENBQUMsY0FBYyxDQUFDLEdBQUcsVUFBVSxDQUFDO1lBQ3JDLFFBQVEsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksR0FBRyxPQUFPLENBQUM7U0FDckM7S0FDRjtJQUVELFNBQVMsSUFBSSxDQUFDLEdBQVcsRUFBRSxHQUFXO1FBQ3BDLENBQUMsQ0FBQyxTQUFTLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLENBQUMsQ0FBQyxXQUFXLEdBQUcsZ0JBQWdCLENBQUM7UUFDakMsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLE9BQU8sRUFBRTtZQUNuQyxDQUFDLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxDQUFDLEdBQUcsR0FBRyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztZQUN2QyxDQUFDLENBQUMsU0FBUyxHQUFHLFNBQVMsQ0FBQztZQUN4QixDQUFDLENBQUMsUUFBUSxDQUFDLFFBQVEsRUFBRSxDQUFDLEdBQUcsR0FBRyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztTQUN2QztRQUNELElBQUksU0FBUyxLQUFLLFNBQVMsQ0FBQyxHQUFHLEVBQUU7WUFDL0IsQ0FBQyxDQUFDLFNBQVMsR0FBRyxLQUFLLENBQUM7WUFDcEIsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxHQUFHLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztTQUN0RjtJQUNILENBQUM7SUFFRCxJQUFJLElBQUksSUFBSSxRQUFRLEdBQUcsQ0FBQyxFQUFFO1FBQ3hCLElBQUksQ0FBQyxNQUFNLEVBQUUsTUFBTSxHQUFHLGdCQUFnQixDQUFDLENBQUM7UUFDeEMsT0FBTyxDQUFDLEdBQUcsZ0JBQWdCLEdBQUcsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxLQUFLLENBQUM7S0FFN0Q7U0FBTTtRQUNMLE1BQU0sRUFBRSxHQUFHLENBQUMsQ0FBQyxHQUFHLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM5QixJQUFJLENBQUMsRUFBRSxFQUFFLEVBQUUsR0FBRyxnQkFBZ0IsQ0FBQyxDQUFDO1FBQ2hDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsR0FBRyxnQkFBZ0IsQ0FBQztLQUNsQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJjb25zdCB1bmRlZmluZWRDb2xvciA9ICdyZ2IoMTAwLDEwMCwxMDApJztcbmNvbnN0IGdyYXlDb2xvciA9ICcjODA4MDgwJztcblxuZXhwb3J0IGVudW0gRHJhd1N0eWxlIHtcbiAgTVNBID0gJ01TQScsXG4gIGNsYXNzaWMgPSAnY2xhc3NpYycsXG59XG5cbi8qKlxuICogQSBmdW5jdGlvbiB0aGF0IHByaW50cyBhIHN0cmluZyBhbGlnbmVkIHRvIGxlZnQgb3IgY2VudGVyZWQuXG4gKlxuICogQHBhcmFtIHtudW1iZXJ9IHggeCBjb29yZGluYXRlLlxuICogQHBhcmFtIHtudW1iZXJ9IHkgeSBjb29yZGluYXRlLlxuICogQHBhcmFtIHtudW1iZXJ9IHcgV2lkdGguXG4gKiBAcGFyYW0ge251bWJlcn0gaCBIZWlnaHQuXG4gKiBAcGFyYW0ge0NhbnZhc1JlbmRlcmluZ0NvbnRleHQyRH0gZyBDYW52YXMgcmVuZGVyaW5nIGNvbnRleHQuXG4gKiBAcGFyYW0ge3N0cmluZ30gcyBTdHJpbmcgdG8gcHJpbnQuXG4gKiBAcGFyYW0ge3N0cmluZ30gW2NvbG9yPXVuZGVmaW5lZENvbG9yXSBTdHJpbmcgY29sb3IuXG4gKiBAcGFyYW0ge251bWJlcn0gW3Bpdm90PTBdIFBpcnZvdC5cbiAqIEBwYXJhbSB7Ym9vbGVhbn0gW2xlZnQ9ZmFsc2VdIElzIGxlZnQgYWxpZ25lZC5cbiAqIEBwYXJhbSB7bnVtYmVyfSBbdHJhbnNwYXJlbmN5UmF0ZT0wLjBdIFRyYW5zcGFyZW5jeSByYXRlIHdoZXJlIDEuMCBpcyBmdWxseSB0cmFuc3BhcmVudFxuICogQHBhcmFtIHtzdHJpbmd9IFtzZXBhcmF0b3I9JyddIElzIHNlcGFyYXRvciBmb3Igc2VxdWVuY2UuXG4gKiBAcGFyYW0ge2Jvb2xlYW59IFtsYXN0PWZhbHNlXSBJcyBjaGVja2VyIGlmIGVsZW1lbnQgbGFzdCBvciBub3QuXG4gKiBAcGFyYW0gZHJhd1N0eWxlIElzIGRyYXcgc3R5bGUuIE1TQSAtIGZvciBtdWx0aWNoYXJTZXEsIGNsYXNzaWMgLSBmb3Igb3RoZXIgc2VxLlxuICogQHBhcmFtIG1heFdvcmQgSXMgYXJyYXkgb2YgbWF4IHdvcmRzIGZvciBlYWNoIGxpbmUuXG4gKiBAcGFyYW0gbWF4V29yZElkeCBJcyBpbmRleCBvZiB3b3JkIHdlIGN1cnJlbnRseSBkcmF3LlxuICogQHBhcmFtIGdyaWRDZWxsIElzIGdyaWQgY2VsbCwgbmV3IGZvciB1cGRhdGluZyBkYXRhIGluIG1heFdvcmQgd2hpbGUgcmVuZGVyaW5nLlxuICogQHJldHVybiB7bnVtYmVyfSB4IGNvb3JkaW5hdGUgdG8gc3RhcnQgcHJpbnRpbmcgYXQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmludExlZnRPckNlbnRlcmVkKFxuICB4OiBudW1iZXIsIHk6IG51bWJlciwgdzogbnVtYmVyLCBoOiBudW1iZXIsXG4gIGc6IENhbnZhc1JlbmRlcmluZ0NvbnRleHQyRCwgczogc3RyaW5nLCBjb2xvciA9IHVuZGVmaW5lZENvbG9yLFxuICBwaXZvdDogbnVtYmVyID0gMCwgbGVmdCA9IGZhbHNlLCB0cmFuc3BhcmVuY3lSYXRlOiBudW1iZXIgPSAxLjAsXG4gIHNlcGFyYXRvcjogc3RyaW5nID0gJycsIGxhc3Q6IGJvb2xlYW4gPSBmYWxzZSwgZHJhd1N0eWxlOiBEcmF3U3R5bGUgPSBEcmF3U3R5bGUuY2xhc3NpYywgbWF4V29yZDogeyBbaW5kZXg6IHN0cmluZ106IG51bWJlciB9ID0ge30sIG1heFdvcmRJZHg6IG51bWJlciA9IDAsIGdyaWRDZWxsOiBhbnkgPSB7fSk6IG51bWJlciB7XG4gIGcudGV4dEFsaWduID0gJ3N0YXJ0JztcbiAgY29uc3QgY29sb3JQYXJ0ID0gcy5zdWJzdHJpbmcoMCk7XG4gIGxldCBncmF5UGFydCA9IGxhc3QgPyAnJyA6IHNlcGFyYXRvcjtcbiAgaWYgKGRyYXdTdHlsZSA9PT0gRHJhd1N0eWxlLk1TQSkge1xuICAgIGdyYXlQYXJ0ID0gJyc7XG4gIH1cblxuICBsZXQgdGV4dFNpemU6IGFueSA9IGcubWVhc3VyZVRleHQoY29sb3JQYXJ0ICsgZ3JheVBhcnQpO1xuICBjb25zdCBpbmRlbnQgPSA1O1xuXG4gIGxldCBtYXhDb2xvclRleHRTaXplID0gZy5tZWFzdXJlVGV4dChjb2xvclBhcnQpLndpZHRoO1xuICBsZXQgY29sb3JUZXh0U2l6ZSA9IGcubWVhc3VyZVRleHQoY29sb3JQYXJ0KS53aWR0aDtcbiAgY29uc3QgZHkgPSAodGV4dFNpemUuZm9udEJvdW5kaW5nQm94QXNjZW50ICsgdGV4dFNpemUuZm9udEJvdW5kaW5nQm94RGVzY2VudCkgLyAyO1xuICB0ZXh0U2l6ZSA9IHRleHRTaXplLndpZHRoO1xuICBpZiAoZHJhd1N0eWxlID09PSBEcmF3U3R5bGUuTVNBKSB7XG4gICAgbWF4Q29sb3JUZXh0U2l6ZSA9IG1heFdvcmRbbWF4V29yZElkeF07XG4gICAgdGV4dFNpemUgPSBtYXhXb3JkW21heFdvcmRJZHhdO1xuICAgIGlmIChtYXhXb3JkSWR4ID4gKG1heFdvcmRbJ2Jpby1tYXhJbmRleCddID8/IDApKSB7XG4gICAgICBtYXhXb3JkWydiaW8tbWF4SW5kZXgnXSA9IG1heFdvcmRJZHg7XG4gICAgICBncmlkQ2VsbC5jZWxsLmNvbHVtbi50ZW1wID0gbWF4V29yZDtcbiAgICB9XG4gIH1cblxuICBmdW5jdGlvbiBkcmF3KGR4MTogbnVtYmVyLCBkeDI6IG51bWJlcik6IHZvaWQge1xuICAgIGcuZmlsbFN0eWxlID0gY29sb3I7XG4gICAgZy5nbG9iYWxBbHBoYSA9IHRyYW5zcGFyZW5jeVJhdGU7XG4gICAgaWYgKGRyYXdTdHlsZSA9PT0gRHJhd1N0eWxlLmNsYXNzaWMpIHtcbiAgICAgIGcuZmlsbFRleHQoY29sb3JQYXJ0LCB4ICsgZHgxLCB5ICsgZHkpO1xuICAgICAgZy5maWxsU3R5bGUgPSBncmF5Q29sb3I7XG4gICAgICBnLmZpbGxUZXh0KGdyYXlQYXJ0LCB4ICsgZHgyLCB5ICsgZHkpO1xuICAgIH1cbiAgICBpZiAoZHJhd1N0eWxlID09PSBEcmF3U3R5bGUuTVNBKSB7XG4gICAgICBnLmZpbGxTdHlsZSA9IGNvbG9yO1xuICAgICAgZy5maWxsVGV4dChjb2xvclBhcnQsIHggKyBkeDEgKyAoKG1heFdvcmRbbWF4V29yZElkeF0gLSBjb2xvclRleHRTaXplKSAvIDIpLCB5ICsgZHkpO1xuICAgIH1cbiAgfVxuXG4gIGlmIChsZWZ0IHx8IHRleHRTaXplID4gdykge1xuICAgIGRyYXcoaW5kZW50LCBpbmRlbnQgKyBtYXhDb2xvclRleHRTaXplKTtcbiAgICByZXR1cm4geCArIG1heENvbG9yVGV4dFNpemUgKyBnLm1lYXN1cmVUZXh0KGdyYXlQYXJ0KS53aWR0aDtcblxuICB9IGVsc2Uge1xuICAgIGNvbnN0IGR4ID0gKHcgLSB0ZXh0U2l6ZSkgLyAyO1xuICAgIGRyYXcoZHgsIGR4ICsgbWF4Q29sb3JUZXh0U2l6ZSk7XG4gICAgcmV0dXJuIHggKyBkeCArIG1heENvbG9yVGV4dFNpemU7XG4gIH1cbn1cblxuIl19","import * as DG from 'datagrok-api/dg';\nimport { UnitsHandler } from './units-handler';\n/** Class for parsing FASTA files */\nexport class FastaFileHandler {\n    constructor(fileContent) {\n        this._descriptionsArray = []; // parsed FASTA descriptions\n        this._sequencesArray = []; // parsed FASTA sequeces\n        this._fileContent = fileContent;\n        this.parseColumns();\n    }\n    // private _columnsParsed: boolean = false;\n    get descriptionsArray() { return this._descriptionsArray; }\n    get sequencesArray() { return this._sequencesArray; }\n    /**\n     * Helper method to parse a macromolecule from a FASTA file (string)\n     *\n     * @param {number} startOfSequence  index of macromolecule substring beginning\n     * @param {number} endOfSequence  index of macromolecule substring end\n  \n     * @return {string} parsed macromolecule\n     */\n    parseMacromolecule(startOfSequence, endOfSequence) {\n        const seq = this._fileContent.slice(startOfSequence, endOfSequence);\n        const seqArray = seq.split(/\\s/);\n        return seqArray.join('');\n    }\n    /** Parse descriptions and sequences from a FASTA string */\n    parseColumns() {\n        const regex = /^>(.*)$/gm; // match 'description' lines starting with >\n        let startOfSequence = 0;\n        let match; // match.index is the beginning of the matched line\n        while (match = regex.exec(this._fileContent)) {\n            const description = this._fileContent.substring(match.index + 1, regex.lastIndex);\n            this._descriptionsArray.push(description);\n            if (startOfSequence !== 0)\n                this._sequencesArray.push(this.parseMacromolecule(startOfSequence, match.index));\n            startOfSequence = regex.lastIndex + 1;\n        }\n        this._sequencesArray.push(this.parseMacromolecule(startOfSequence, -1));\n        // this._columnsParsed = true;\n    }\n    /**\n     * File-handler method for import as FASTA\n     *\n     * @return {DG.DataFrame[]} dataframe with parsed FASTA content\n     */\n    importFasta() {\n        const descriptionsArrayCol = DG.Column.fromStrings('description', this.descriptionsArray);\n        const sequenceCol = DG.Column.fromStrings('sequence', this.sequencesArray);\n        sequenceCol.semType = DG.SEMTYPE.MACROMOLECULE;\n        // here should go the code from units handler\n        UnitsHandler.setUnitsToFastaColumn(sequenceCol);\n        return [DG.DataFrame.fromColumns([\n                descriptionsArrayCol,\n                sequenceCol,\n            ])];\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFzdGEtaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImZhc3RhLWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0EsT0FBTyxLQUFLLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUV0QyxPQUFPLEVBQUMsWUFBWSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFFN0Msb0NBQW9DO0FBQ3BDLE1BQU0sT0FBTyxnQkFBZ0I7SUFnRTNCLFlBQVksV0FBbUI7UUE5RHZCLHVCQUFrQixHQUFhLEVBQUUsQ0FBQyxDQUFDLDRCQUE0QjtRQUMvRCxvQkFBZSxHQUFhLEVBQUUsQ0FBQyxDQUFDLHdCQUF3QjtRQThEOUQsSUFBSSxDQUFDLFlBQVksR0FBRyxXQUFXLENBQUM7UUFDaEMsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUEvREQsMkNBQTJDO0lBRTNDLElBQVcsaUJBQWlCLEtBQWUsT0FBTyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQyxDQUFDO0lBRTVFLElBQVcsY0FBYyxLQUFlLE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxDQUFDLENBQUM7SUFFdEU7Ozs7Ozs7T0FPRztJQUNLLGtCQUFrQixDQUN4QixlQUF1QixFQUN2QixhQUFxQjtRQUVyQixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxlQUFlLEVBQUUsYUFBYSxDQUFDLENBQUM7UUFDcEUsTUFBTSxRQUFRLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNqQyxPQUFPLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDM0IsQ0FBQztJQUVELDJEQUEyRDtJQUNuRCxZQUFZO1FBQ2xCLE1BQU0sS0FBSyxHQUFHLFdBQVcsQ0FBQyxDQUFDLDRDQUE0QztRQUV2RSxJQUFJLGVBQWUsR0FBRyxDQUFDLENBQUM7UUFDeEIsSUFBSSxLQUFLLENBQUMsQ0FBQyxtREFBbUQ7UUFDOUQsT0FBTyxLQUFLLEdBQUcsS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUU7WUFDNUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEtBQUssR0FBRyxDQUFDLEVBQUUsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ2xGLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7WUFDMUMsSUFBSSxlQUFlLEtBQUssQ0FBQztnQkFDdkIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGtCQUFrQixDQUFDLGVBQWUsRUFBRSxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztZQUNuRixlQUFlLEdBQUcsS0FBSyxDQUFDLFNBQVMsR0FBRyxDQUFDLENBQUM7U0FDdkM7UUFDRCxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsZUFBZSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUV4RSw4QkFBOEI7SUFDaEMsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxXQUFXO1FBQ2hCLE1BQU0sb0JBQW9CLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUMsYUFBYSxFQUFFLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQzFGLE1BQU0sV0FBVyxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUM7UUFDM0UsV0FBVyxDQUFDLE9BQU8sR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQztRQUUvQyw2Q0FBNkM7UUFDN0MsWUFBWSxDQUFDLHFCQUFxQixDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRWhELE9BQU8sQ0FBQyxFQUFFLENBQUMsU0FBUyxDQUFDLFdBQVcsQ0FBQztnQkFDL0Isb0JBQW9CO2dCQUNwQixXQUFXO2FBQ1osQ0FBQyxDQUFDLENBQUM7SUFDTixDQUFDO0NBTUYiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBEbyBub3QgY2hhbmdlIHRoZXNlIGltcG9ydCBsaW5lcyB0byBtYXRjaCBleHRlcm5hbCBtb2R1bGVzIGluIHdlYnBhY2sgY29uZmlndXJhdGlvbiAqL1xuaW1wb3J0ICogYXMgZ3JvayBmcm9tICdkYXRhZ3Jvay1hcGkvZ3Jvayc7XG5pbXBvcnQgKiBhcyB1aSBmcm9tICdkYXRhZ3Jvay1hcGkvdWknO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuaW1wb3J0IHtVbml0c0hhbmRsZXJ9IGZyb20gJy4vdW5pdHMtaGFuZGxlcic7XG5cbi8qKiBDbGFzcyBmb3IgcGFyc2luZyBGQVNUQSBmaWxlcyAqL1xuZXhwb3J0IGNsYXNzIEZhc3RhRmlsZUhhbmRsZXIge1xuICBwcml2YXRlIF9maWxlQ29udGVudDogc3RyaW5nO1xuICBwcml2YXRlIF9kZXNjcmlwdGlvbnNBcnJheTogc3RyaW5nW10gPSBbXTsgLy8gcGFyc2VkIEZBU1RBIGRlc2NyaXB0aW9uc1xuICBwcml2YXRlIF9zZXF1ZW5jZXNBcnJheTogc3RyaW5nW10gPSBbXTsgLy8gcGFyc2VkIEZBU1RBIHNlcXVlY2VzXG4gIC8vIHByaXZhdGUgX2NvbHVtbnNQYXJzZWQ6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBwdWJsaWMgZ2V0IGRlc2NyaXB0aW9uc0FycmF5KCk6IHN0cmluZ1tdIHsgcmV0dXJuIHRoaXMuX2Rlc2NyaXB0aW9uc0FycmF5OyB9XG5cbiAgcHVibGljIGdldCBzZXF1ZW5jZXNBcnJheSgpOiBzdHJpbmdbXSB7IHJldHVybiB0aGlzLl9zZXF1ZW5jZXNBcnJheTsgfVxuXG4gIC8qKlxuICAgKiBIZWxwZXIgbWV0aG9kIHRvIHBhcnNlIGEgbWFjcm9tb2xlY3VsZSBmcm9tIGEgRkFTVEEgZmlsZSAoc3RyaW5nKVxuICAgKlxuICAgKiBAcGFyYW0ge251bWJlcn0gc3RhcnRPZlNlcXVlbmNlICBpbmRleCBvZiBtYWNyb21vbGVjdWxlIHN1YnN0cmluZyBiZWdpbm5pbmdcbiAgICogQHBhcmFtIHtudW1iZXJ9IGVuZE9mU2VxdWVuY2UgIGluZGV4IG9mIG1hY3JvbW9sZWN1bGUgc3Vic3RyaW5nIGVuZFxuXG4gICAqIEByZXR1cm4ge3N0cmluZ30gcGFyc2VkIG1hY3JvbW9sZWN1bGVcbiAgICovXG4gIHByaXZhdGUgcGFyc2VNYWNyb21vbGVjdWxlKFxuICAgIHN0YXJ0T2ZTZXF1ZW5jZTogbnVtYmVyLFxuICAgIGVuZE9mU2VxdWVuY2U6IG51bWJlclxuICApOiBzdHJpbmcge1xuICAgIGNvbnN0IHNlcSA9IHRoaXMuX2ZpbGVDb250ZW50LnNsaWNlKHN0YXJ0T2ZTZXF1ZW5jZSwgZW5kT2ZTZXF1ZW5jZSk7XG4gICAgY29uc3Qgc2VxQXJyYXkgPSBzZXEuc3BsaXQoL1xccy8pO1xuICAgIHJldHVybiBzZXFBcnJheS5qb2luKCcnKTtcbiAgfVxuXG4gIC8qKiBQYXJzZSBkZXNjcmlwdGlvbnMgYW5kIHNlcXVlbmNlcyBmcm9tIGEgRkFTVEEgc3RyaW5nICovXG4gIHByaXZhdGUgcGFyc2VDb2x1bW5zKCkge1xuICAgIGNvbnN0IHJlZ2V4ID0gL14+KC4qKSQvZ207IC8vIG1hdGNoICdkZXNjcmlwdGlvbicgbGluZXMgc3RhcnRpbmcgd2l0aCA+XG5cbiAgICBsZXQgc3RhcnRPZlNlcXVlbmNlID0gMDtcbiAgICBsZXQgbWF0Y2g7IC8vIG1hdGNoLmluZGV4IGlzIHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1hdGNoZWQgbGluZVxuICAgIHdoaWxlIChtYXRjaCA9IHJlZ2V4LmV4ZWModGhpcy5fZmlsZUNvbnRlbnQpKSB7XG4gICAgICBjb25zdCBkZXNjcmlwdGlvbiA9IHRoaXMuX2ZpbGVDb250ZW50LnN1YnN0cmluZyhtYXRjaC5pbmRleCArIDEsIHJlZ2V4Lmxhc3RJbmRleCk7XG4gICAgICB0aGlzLl9kZXNjcmlwdGlvbnNBcnJheS5wdXNoKGRlc2NyaXB0aW9uKTtcbiAgICAgIGlmIChzdGFydE9mU2VxdWVuY2UgIT09IDApXG4gICAgICAgIHRoaXMuX3NlcXVlbmNlc0FycmF5LnB1c2godGhpcy5wYXJzZU1hY3JvbW9sZWN1bGUoc3RhcnRPZlNlcXVlbmNlLCBtYXRjaC5pbmRleCkpO1xuICAgICAgc3RhcnRPZlNlcXVlbmNlID0gcmVnZXgubGFzdEluZGV4ICsgMTtcbiAgICB9XG4gICAgdGhpcy5fc2VxdWVuY2VzQXJyYXkucHVzaCh0aGlzLnBhcnNlTWFjcm9tb2xlY3VsZShzdGFydE9mU2VxdWVuY2UsIC0xKSk7XG5cbiAgICAvLyB0aGlzLl9jb2x1bW5zUGFyc2VkID0gdHJ1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBGaWxlLWhhbmRsZXIgbWV0aG9kIGZvciBpbXBvcnQgYXMgRkFTVEFcbiAgICpcbiAgICogQHJldHVybiB7REcuRGF0YUZyYW1lW119IGRhdGFmcmFtZSB3aXRoIHBhcnNlZCBGQVNUQSBjb250ZW50XG4gICAqL1xuICBwdWJsaWMgaW1wb3J0RmFzdGEoKTogREcuRGF0YUZyYW1lIFtdIHtcbiAgICBjb25zdCBkZXNjcmlwdGlvbnNBcnJheUNvbCA9IERHLkNvbHVtbi5mcm9tU3RyaW5ncygnZGVzY3JpcHRpb24nLCB0aGlzLmRlc2NyaXB0aW9uc0FycmF5KTtcbiAgICBjb25zdCBzZXF1ZW5jZUNvbCA9IERHLkNvbHVtbi5mcm9tU3RyaW5ncygnc2VxdWVuY2UnLCB0aGlzLnNlcXVlbmNlc0FycmF5KTtcbiAgICBzZXF1ZW5jZUNvbC5zZW1UeXBlID0gREcuU0VNVFlQRS5NQUNST01PTEVDVUxFO1xuXG4gICAgLy8gaGVyZSBzaG91bGQgZ28gdGhlIGNvZGUgZnJvbSB1bml0cyBoYW5kbGVyXG4gICAgVW5pdHNIYW5kbGVyLnNldFVuaXRzVG9GYXN0YUNvbHVtbihzZXF1ZW5jZUNvbCk7XG5cbiAgICByZXR1cm4gW0RHLkRhdGFGcmFtZS5mcm9tQ29sdW1ucyhbXG4gICAgICBkZXNjcmlwdGlvbnNBcnJheUNvbCxcbiAgICAgIHNlcXVlbmNlQ29sLFxuICAgIF0pXTtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKGZpbGVDb250ZW50OiBzdHJpbmcpIHtcbiAgICB0aGlzLl9maWxlQ29udGVudCA9IGZpbGVDb250ZW50O1xuICAgIHRoaXMucGFyc2VDb2x1bW5zKCk7XG4gIH1cbn1cbiJdfQ==","import { WebLogo } from '../viewers/web-logo';\nimport { UnitsHandler } from './units-handler';\n/** Class for handling conversion of notation systems in Macromolecule columns */\nexport class NotationConverter extends UnitsHandler {\n    constructor(col) {\n        super(col);\n        this._splitter = null;\n    }\n    get splitter() {\n        if (this._splitter === null)\n            this._splitter = WebLogo.getSplitterForColumn(this.column);\n        return this._splitter;\n    }\n    ;\n    toFasta(targetNotation) { return targetNotation === \"fasta\" /* NOTATION.FASTA */; }\n    toSeparator(targetNotation) { return targetNotation === \"separator\" /* NOTATION.SEPARATOR */; }\n    toHelm(targetNotation) { return targetNotation === \"helm\" /* NOTATION.HELM */; }\n    /**\n     * Convert a Macromolecule column from FASTA to SEPARATOR notation\n     *\n     * @param {string} separator  A specific separator to be used\n     * @param {string} fastaGapSymbol  Gap symbol in FASTA, '-' by default\n     * @return {DG.Column}        A new column in SEPARATOR notation\n     */\n    convertFastaToSeparator(separator, fastaGapSymbol = null) {\n        if (fastaGapSymbol === null)\n            fastaGapSymbol = this.defaultGapSymbol;\n        const newColumn = this.getNewColumn(\"separator\" /* NOTATION.SEPARATOR */);\n        // assign the values to the newly created empty column\n        newColumn.init((idx) => {\n            const fastaPolymer = this.column.get(idx);\n            const fastaMonomersArray = this.splitter(fastaPolymer);\n            for (let i = 0; i < fastaMonomersArray.length; i++) {\n                if (fastaMonomersArray[i] === fastaGapSymbol)\n                    fastaMonomersArray[i] = UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n            }\n            return fastaMonomersArray.join(separator);\n        });\n        newColumn.setTag(UnitsHandler.TAGS.separator, separator);\n        return newColumn;\n    }\n    /**\n     * Get the wrapper strings for HELM, depending on the type of the\n     * macromolecule (peptide, DNA, RNA)\n     *\n     * @return {string[]} Array of wrappers\n     */\n    getHelmWrappers() {\n        const prefix = (this.isDna()) ? 'DNA1{' :\n            (this.isRna()) ? 'RNA1{' :\n                (this.isPeptide()) ? 'PEPTIDE1{' :\n                    'Unknown'; // this case should be handled as exceptional\n        if (prefix === 'Unknown')\n            throw new Error('Neither peptide, nor nucleotide');\n        const postfix = '}$$$';\n        const leftWrapper = (this.isDna()) ? 'D(' :\n            (this.isRna()) ? 'R(' : ''; // no wrapper for peptides\n        const rightWrapper = (this.isDna() || this.isRna()) ? ')P' : ''; // no wrapper for peptides\n        return [prefix, leftWrapper, rightWrapper, postfix];\n    }\n    // A helper function for converting strings to HELM\n    convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix) {\n        const monomerArray = this.splitter(sourcePolymer);\n        const monomerHelmArray = monomerArray.map((mm) => {\n            if (mm === sourceGapSymbol)\n                return UnitsHandler._defaultGapSymbolsDict.HELM;\n            else\n                return `${leftWrapper}${mm}${rightWrapper}`;\n        });\n        return `${prefix}${monomerHelmArray.join('.')}${postfix}`;\n    }\n    /**\n     * Convert a string with SEPARATOR/FASTA notation to HELM\n     *\n     * @param {string} sourcePolymer  A string to be converted\n     * @param {string | null} sourceGapSymbol  An optional gap symbol, set to\n     * default values ('-' for FASTA and '' for SEPARATOR) unless specified\n     * @return {string}  The target HELM string\n     */\n    convertStringToHelm(sourcePolymer, sourceGapSymbol = null) {\n        if (sourceGapSymbol === null)\n            sourceGapSymbol = this.defaultGapSymbol;\n        const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n        return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n    }\n    /**\n     * Convert a column to HELM\n     *\n     * @param {string | null} sourceGapSymbol\n     * @return {DG.Column}\n     */\n    convertToHelm(sourceGapSymbol = null) {\n        if (sourceGapSymbol === null)\n            sourceGapSymbol = this.defaultGapSymbol;\n        const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n        const newColumn = this.getNewColumn(\"helm\" /* NOTATION.HELM */);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const sourcePolymer = this.column.get(idx);\n            return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n        });\n        return newColumn;\n    }\n    /**\n     * Convert SEPARATOR column to FASTA notation\n     *\n     * @param {string | null} fastaGapSymbol Optional gap symbol for FASTA\n     * @return {DG.Column}  Converted column\n     */\n    convertSeparatorToFasta(fastaGapSymbol = null) {\n        if (fastaGapSymbol === null)\n            fastaGapSymbol = UnitsHandler._defaultGapSymbolsDict.FASTA;\n        const newColumn = this.getNewColumn(\"fasta\" /* NOTATION.FASTA */);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const separatorPolymer = this.column.get(idx);\n            // items can be monomers or separators\n            const separatorItemsArray = this.splitter(separatorPolymer);\n            const fastaMonomersArray = [];\n            for (let i = 0; i < separatorItemsArray.length; i++) {\n                const item = separatorItemsArray[i];\n                if (item.length === 0) {\n                    fastaMonomersArray.push(fastaGapSymbol);\n                }\n                else if (item.length > 1) {\n                    // the case of a multi-character monomer\n                    const monomer = '[' + item + ']';\n                    fastaMonomersArray.push(monomer);\n                }\n                else {\n                    fastaMonomersArray.push(item);\n                }\n            }\n            return fastaMonomersArray.join('');\n        });\n        return newColumn;\n    }\n    /**\n     *  Convert HELM column to FASTA/SEPARATOR\n     *\n     * @param {string} tgtNotation    Target notation: FASTA or SEPARATOR\n     * @param {string} tgtSeparator   Optional target separator (for HELM ->\n     * @param {string | null} tgtGapSymbol   Optional target gap symbol\n     * SEPARATOR)\n     * @return {DG.Column} Converted column\n     */\n    convertHelm(tgtNotation, tgtSeparator = '', tgtGapSymbol = null) {\n        // This function must not contain calls of isDna() and isRna(), for\n        // source helm columns may contain RNA, DNA and PT across different rows\n        if (tgtGapSymbol === null) {\n            tgtGapSymbol = (this.toFasta(tgtNotation)) ?\n                UnitsHandler._defaultGapSymbolsDict.FASTA :\n                UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n        }\n        if (this.toSeparator(tgtNotation) && tgtSeparator === '')\n            tgtSeparator = this.separator;\n        const helmWrappersRe = /(R\\(|D\\(|\\)|P)/g;\n        const newColumn = this.getNewColumn(tgtNotation);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const helmPolymer = this.column.get(idx);\n            // we cannot use isDna() or isRna() because source helm columns can\n            // contain DNA, RNA and PT in different cells, so the corresponding\n            // tags cannot be set for the whole column\n            const isNucleotide = helmPolymer.startsWith('DNA') || helmPolymer.startsWith('RNA');\n            // items can be monomers or helms\n            const helmItemsArray = this.splitter(helmPolymer);\n            const tgtMonomersArray = [];\n            for (let i = 0; i < helmItemsArray.length; i++) {\n                let item = helmItemsArray[i];\n                if (isNucleotide)\n                    item = item.replace(helmWrappersRe, '');\n                if (item === UnitsHandler._defaultGapSymbolsDict.HELM) {\n                    tgtMonomersArray.push(tgtGapSymbol);\n                }\n                else if (this.toFasta(tgtNotation) && item.length > 1) {\n                    // the case of a multi-character monomer converted to FASTA\n                    const monomer = '[' + item + ']';\n                    tgtMonomersArray.push(monomer);\n                }\n                else {\n                    tgtMonomersArray.push(item);\n                }\n            }\n            return tgtMonomersArray.join(tgtSeparator);\n        });\n        return newColumn;\n    }\n    convertHelmToSeparator() {\n        // TODO: implementatioreturn this.getNewColumn();\n        return this.getNewColumn(\"separator\" /* NOTATION.SEPARATOR */);\n    }\n    /** Dispatcher method for notation conversion\n     *\n     * @param {NOTATION} tgtNotation   Notation we want to convert to\n     * @param {string | null} tgtSeparator   Possible separator\n     * @return {DG.Column}                Converted column\n     */\n    convert(tgtNotation, tgtSeparator = null) {\n        // possible exceptions\n        if (this.notation === tgtNotation)\n            throw new Error('tgt notation is invalid');\n        if (this.toSeparator(tgtNotation) && tgtSeparator === null)\n            throw new Error('tgt separator is not specified');\n        if (this.isFasta() && this.toSeparator(tgtNotation) && tgtSeparator !== null)\n            return this.convertFastaToSeparator(tgtSeparator);\n        else if ((this.isFasta() || this.isSeparator()) && this.toHelm(tgtNotation))\n            return this.convertToHelm();\n        else if (this.isSeparator() && this.toFasta(tgtNotation))\n            return this.convertSeparatorToFasta();\n        else if (this.isHelm() && this.toFasta(tgtNotation)) // the case of HELM\n            return this.convertHelm(tgtNotation);\n        else // this.isHelm() && this.toSeparator(tgtNotation)\n            return this.convertHelm(tgtNotation, tgtSeparator);\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notation-converter.js","sourceRoot":"","sources":["notation-converter.ts"],"names":[],"mappings":"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","sourcesContent":["/* Do not change these import lines to match external modules in webpack configuration */\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport {SplitterFunc, WebLogo} from '../viewers/web-logo';\nimport {UnitsHandler, NOTATION} from './units-handler';\n\n/** Class for handling conversion of notation systems in Macromolecule columns */\nexport class NotationConverter extends UnitsHandler {\n  private _splitter: SplitterFunc | null = null;\n  protected get splitter(): SplitterFunc {\n    if (this._splitter === null)\n      this._splitter = WebLogo.getSplitterForColumn(this.column);\n    return this._splitter;\n  };\n\n  public toFasta(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.FASTA; }\n\n  public toSeparator(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.SEPARATOR; }\n\n  public toHelm(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.HELM; }\n\n  /**\n   * Convert a Macromolecule column from FASTA to SEPARATOR notation\n   *\n   * @param {string} separator  A specific separator to be used\n   * @param {string} fastaGapSymbol  Gap symbol in FASTA, '-' by default\n   * @return {DG.Column}        A new column in SEPARATOR notation\n   */\n  private convertFastaToSeparator(separator: string, fastaGapSymbol: string | null = null): DG.Column {\n    if (fastaGapSymbol === null)\n      fastaGapSymbol = this.defaultGapSymbol;\n\n    const newColumn = this.getNewColumn(NOTATION.SEPARATOR);\n    // assign the values to the newly created empty column\n    newColumn.init((idx: number) => {\n      const fastaPolymer = this.column.get(idx);\n      const fastaMonomersArray = this.splitter(fastaPolymer);\n      for (let i = 0; i < fastaMonomersArray.length; i++) {\n        if (fastaMonomersArray[i] === fastaGapSymbol)\n          fastaMonomersArray[i] = UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n      }\n      return fastaMonomersArray.join(separator);\n    });\n    newColumn.setTag(UnitsHandler.TAGS.separator, separator);\n    return newColumn;\n  }\n\n  /**\n   * Get the wrapper strings for HELM, depending on the type of the\n   * macromolecule (peptide, DNA, RNA)\n   *\n   * @return {string[]} Array of wrappers\n   */\n  private getHelmWrappers(): string[] {\n    const prefix = (this.isDna()) ? 'DNA1{' :\n      (this.isRna()) ? 'RNA1{' :\n        (this.isPeptide()) ? 'PEPTIDE1{' :\n          'Unknown'; // this case should be handled as exceptional\n\n    if (prefix === 'Unknown')\n      throw new Error('Neither peptide, nor nucleotide');\n\n    const postfix = '}$$$';\n    const leftWrapper = (this.isDna()) ? 'D(' :\n      (this.isRna()) ? 'R(' : ''; // no wrapper for peptides\n    const rightWrapper = (this.isDna() || this.isRna()) ? ')P' : ''; // no wrapper for peptides\n    return [prefix, leftWrapper, rightWrapper, postfix];\n  }\n\n  // A helper function for converting strings to HELM\n  private convertToHelmHelper(\n    sourcePolymer: string,\n    sourceGapSymbol: string,\n    prefix: string,\n    leftWrapper: string,\n    rightWrapper: string,\n    postfix: string\n  ): string {\n    const monomerArray = this.splitter(sourcePolymer);\n    const monomerHelmArray: string[] = monomerArray.map((mm: string) => {\n      if (mm === sourceGapSymbol)\n        return UnitsHandler._defaultGapSymbolsDict.HELM;\n      else\n        return `${leftWrapper}${mm}${rightWrapper}`;\n    });\n    return `${prefix}${monomerHelmArray.join('.')}${postfix}`;\n  }\n\n  /**\n   * Convert a string with SEPARATOR/FASTA notation to HELM\n   *\n   * @param {string} sourcePolymer  A string to be converted\n   * @param {string | null} sourceGapSymbol  An optional gap symbol, set to\n   * default values ('-' for FASTA and '' for SEPARATOR) unless specified\n   * @return {string}  The target HELM string\n   */\n  public convertStringToHelm(\n    sourcePolymer: string,\n    sourceGapSymbol: string | null = null\n  ): string {\n    if (sourceGapSymbol === null)\n      sourceGapSymbol = this.defaultGapSymbol;\n    const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n    return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n  }\n\n  /**\n   * Convert a column to HELM\n   *\n   * @param {string | null} sourceGapSymbol\n   * @return {DG.Column}\n   */\n  private convertToHelm(sourceGapSymbol: string | null = null): DG.Column {\n    if (sourceGapSymbol === null)\n      sourceGapSymbol = this.defaultGapSymbol;\n\n    const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n\n    const newColumn = this.getNewColumn(NOTATION.HELM);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const sourcePolymer = this.column.get(idx);\n      return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol!, prefix, leftWrapper, rightWrapper, postfix);\n    });\n    return newColumn;\n  }\n\n  /**\n   * Convert SEPARATOR column to FASTA notation\n   *\n   * @param {string | null} fastaGapSymbol Optional gap symbol for FASTA\n   * @return {DG.Column}  Converted column\n   */\n  private convertSeparatorToFasta(fastaGapSymbol: string | null = null): DG.Column {\n    if (fastaGapSymbol === null)\n      fastaGapSymbol = UnitsHandler._defaultGapSymbolsDict.FASTA;\n\n    const newColumn = this.getNewColumn(NOTATION.FASTA);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const separatorPolymer = this.column.get(idx);\n      // items can be monomers or separators\n      const separatorItemsArray = this.splitter(separatorPolymer);\n      const fastaMonomersArray: string[] = [];\n      for (let i = 0; i < separatorItemsArray.length; i++) {\n        const item = separatorItemsArray[i];\n        if (item.length === 0) {\n          fastaMonomersArray.push(fastaGapSymbol!);\n        } else if (item.length > 1) {\n          // the case of a multi-character monomer\n          const monomer = '[' + item + ']';\n          fastaMonomersArray.push(monomer);\n        } else {\n          fastaMonomersArray.push(item);\n        }\n      }\n      return fastaMonomersArray.join('');\n    });\n    return newColumn;\n  }\n\n  /**\n   *  Convert HELM column to FASTA/SEPARATOR\n   *\n   * @param {string} tgtNotation    Target notation: FASTA or SEPARATOR\n   * @param {string} tgtSeparator   Optional target separator (for HELM ->\n   * @param {string | null} tgtGapSymbol   Optional target gap symbol\n   * SEPARATOR)\n   * @return {DG.Column} Converted column\n   */\n  private convertHelm(\n    tgtNotation: string,\n    tgtSeparator: string = '',\n    tgtGapSymbol: string | null = null\n  ): DG.Column {\n    // This function must not contain calls of isDna() and isRna(), for\n    // source helm columns may contain RNA, DNA and PT across different rows\n    if (tgtGapSymbol === null) {\n      tgtGapSymbol = (this.toFasta(tgtNotation as NOTATION)) ?\n        UnitsHandler._defaultGapSymbolsDict.FASTA :\n        UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n    }\n\n    if (this.toSeparator(tgtNotation as NOTATION) && tgtSeparator === '')\n      tgtSeparator = this.separator;\n\n    const helmWrappersRe = /(R\\(|D\\(|\\)|P)/g;\n    const newColumn = this.getNewColumn(tgtNotation as NOTATION);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const helmPolymer = this.column.get(idx);\n\n      // we cannot use isDna() or isRna() because source helm columns can\n      // contain DNA, RNA and PT in different cells, so the corresponding\n      // tags cannot be set for the whole column\n      const isNucleotide = helmPolymer.startsWith('DNA') || helmPolymer.startsWith('RNA');\n\n      // items can be monomers or helms\n      const helmItemsArray = this.splitter(helmPolymer);\n      const tgtMonomersArray: string[] = [];\n      for (let i = 0; i < helmItemsArray.length; i++) {\n        let item = helmItemsArray[i];\n        if (isNucleotide)\n          item = item.replace(helmWrappersRe, '');\n        if (item === UnitsHandler._defaultGapSymbolsDict.HELM) {\n          tgtMonomersArray.push(tgtGapSymbol!);\n        } else if (this.toFasta(tgtNotation as NOTATION) && item.length > 1) {\n          // the case of a multi-character monomer converted to FASTA\n          const monomer = '[' + item + ']';\n          tgtMonomersArray.push(monomer);\n        } else {\n          tgtMonomersArray.push(item);\n        }\n      }\n      return tgtMonomersArray.join(tgtSeparator);\n    });\n    return newColumn;\n  }\n\n  private convertHelmToSeparator(): DG.Column {\n    // TODO: implementatioreturn this.getNewColumn();\n    return this.getNewColumn(NOTATION.SEPARATOR);\n  }\n\n  /** Dispatcher method for notation conversion\n   *\n   * @param {NOTATION} tgtNotation   Notation we want to convert to\n   * @param {string | null} tgtSeparator   Possible separator\n   * @return {DG.Column}                Converted column\n   */\n  public convert(tgtNotation: NOTATION, tgtSeparator: string | null = null): DG.Column {\n    // possible exceptions\n    if (this.notation === tgtNotation)\n      throw new Error('tgt notation is invalid');\n    if (this.toSeparator(tgtNotation) && tgtSeparator === null)\n      throw new Error('tgt separator is not specified');\n\n    if (this.isFasta() && this.toSeparator(tgtNotation) && tgtSeparator !== null)\n      return this.convertFastaToSeparator(tgtSeparator);\n    else if ((this.isFasta() || this.isSeparator()) && this.toHelm(tgtNotation))\n      return this.convertToHelm();\n    else if (this.isSeparator() && this.toFasta(tgtNotation))\n      return this.convertSeparatorToFasta();\n    else if (this.isHelm() && this.toFasta(tgtNotation)) // the case of HELM\n      return this.convertHelm(tgtNotation);\n    else // this.isHelm() && this.toSeparator(tgtNotation)\n      return this.convertHelm(tgtNotation, tgtSeparator!);\n  }\n\n  public constructor(col: DG.Column) {\n    super(col);\n  }\n}\n"]}","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '../viewers/web-logo';\nexport function splitAlignedSequences(peptideColumn) {\n    const splitter = WebLogo.getSplitterForColumn(peptideColumn);\n    const colLen = peptideColumn.length;\n    const resultDf = DG.DataFrame.create(colLen);\n    let monomerList = splitter(peptideColumn.get(0));\n    const columnList = [];\n    // create columns and fill the first row for faster values filling in the next loop\n    for (let i = 0; i < monomerList.length; i++) {\n        const col = resultDf.columns.addNewString((i + 1).toString());\n        col.set(0, monomerList[i] || '-', false);\n        columnList.push(col);\n    }\n    for (let rowIndex = 1; rowIndex < colLen; rowIndex++) {\n        monomerList = splitter(peptideColumn.get(rowIndex));\n        monomerList.forEach((monomer, colIndex) => columnList[colIndex].set(rowIndex, monomer || '-', false));\n    }\n    return resultDf;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BsaXR0ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzcGxpdHRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRXRDLE9BQU8sRUFBQyxPQUFPLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUU1QyxNQUFNLFVBQVUscUJBQXFCLENBQUMsYUFBZ0M7SUFDcEUsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLG9CQUFvQixDQUFDLGFBQWEsQ0FBQyxDQUFDO0lBQzdELE1BQU0sTUFBTSxHQUFHLGFBQWEsQ0FBQyxNQUFNLENBQUM7SUFDcEMsTUFBTSxRQUFRLEdBQUcsRUFBRSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDN0MsSUFBSSxXQUFXLEdBQUcsUUFBUSxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFFLENBQUMsQ0FBQztJQUNsRCxNQUFNLFVBQVUsR0FBd0IsRUFBRSxDQUFDO0lBRTNDLG1GQUFtRjtJQUNuRixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsV0FBVyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtRQUMzQyxNQUFNLEdBQUcsR0FBRyxRQUFRLENBQUMsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQzlELEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsSUFBSSxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDekMsVUFBVSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztLQUN0QjtJQUVELEtBQUssSUFBSSxRQUFRLEdBQUcsQ0FBQyxFQUFFLFFBQVEsR0FBRyxNQUFNLEVBQUUsUUFBUSxFQUFFLEVBQUU7UUFDcEQsV0FBVyxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBRSxDQUFDLENBQUM7UUFDckQsV0FBVyxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsRUFBRSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxHQUFHLENBQUMsUUFBUSxFQUFFLE9BQU8sSUFBSSxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztLQUN2RztJQUVELE9BQU8sUUFBUSxDQUFDO0FBQ2xCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG5pbXBvcnQge1dlYkxvZ299IGZyb20gJy4uL3ZpZXdlcnMvd2ViLWxvZ28nO1xuXG5leHBvcnQgZnVuY3Rpb24gc3BsaXRBbGlnbmVkU2VxdWVuY2VzKHBlcHRpZGVDb2x1bW46IERHLkNvbHVtbjxzdHJpbmc+KTogREcuRGF0YUZyYW1lIHtcbiAgY29uc3Qgc3BsaXR0ZXIgPSBXZWJMb2dvLmdldFNwbGl0dGVyRm9yQ29sdW1uKHBlcHRpZGVDb2x1bW4pO1xuICBjb25zdCBjb2xMZW4gPSBwZXB0aWRlQ29sdW1uLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0RGYgPSBERy5EYXRhRnJhbWUuY3JlYXRlKGNvbExlbik7XG4gIGxldCBtb25vbWVyTGlzdCA9IHNwbGl0dGVyKHBlcHRpZGVDb2x1bW4uZ2V0KDApISk7XG4gIGNvbnN0IGNvbHVtbkxpc3Q6IERHLkNvbHVtbjxzdHJpbmc+W10gPSBbXTtcblxuICAvLyBjcmVhdGUgY29sdW1ucyBhbmQgZmlsbCB0aGUgZmlyc3Qgcm93IGZvciBmYXN0ZXIgdmFsdWVzIGZpbGxpbmcgaW4gdGhlIG5leHQgbG9vcFxuICBmb3IgKGxldCBpID0gMDsgaSA8IG1vbm9tZXJMaXN0Lmxlbmd0aDsgaSsrKSB7XG4gICAgY29uc3QgY29sID0gcmVzdWx0RGYuY29sdW1ucy5hZGROZXdTdHJpbmcoKGkgKyAxKS50b1N0cmluZygpKTtcbiAgICBjb2wuc2V0KDAsIG1vbm9tZXJMaXN0W2ldIHx8ICctJywgZmFsc2UpO1xuICAgIGNvbHVtbkxpc3QucHVzaChjb2wpO1xuICB9XG5cbiAgZm9yIChsZXQgcm93SW5kZXggPSAxOyByb3dJbmRleCA8IGNvbExlbjsgcm93SW5kZXgrKykge1xuICAgIG1vbm9tZXJMaXN0ID0gc3BsaXR0ZXIocGVwdGlkZUNvbHVtbi5nZXQocm93SW5kZXgpISk7XG4gICAgbW9ub21lckxpc3QuZm9yRWFjaCgobW9ub21lciwgY29sSW5kZXgpID0+IGNvbHVtbkxpc3RbY29sSW5kZXhdLnNldChyb3dJbmRleCwgbW9ub21lciB8fCAnLScsIGZhbHNlKSk7XG4gIH1cblxuICByZXR1cm4gcmVzdWx0RGY7XG59XG4gICJdfQ==","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '../viewers/web-logo';\n/** Class for handling notation units in Macromolecule columns */\nexport class UnitsHandler {\n    constructor(col) {\n        this._column = col;\n        const units = this._column.tags[DG.TAGS.UNITS];\n        if (units !== null)\n            this._units = units;\n        else\n            throw new Error('Units are not specified in column');\n        this._notation = this.getNotation();\n        this._defaultGapSymbol = (this.isFasta()) ? UnitsHandler._defaultGapSymbolsDict.FASTA :\n            (this.isHelm()) ? UnitsHandler._defaultGapSymbolsDict.HELM :\n                UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n        if (!this.column.tags.has(UnitsHandler.TAGS.alphabetSize)) {\n            if (this.isHelm())\n                throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n            else if (['UN'].includes(this.alphabet))\n                throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n        }\n        if (!this.column.tags.has(UnitsHandler.TAGS.alphabetIsMultichar)) {\n            if (this.isHelm())\n                throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n            else if (['UN'].includes(this.alphabet))\n                throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n        }\n    }\n    static setUnitsToFastaColumn(col) {\n        if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n            throw new Error('Fasta column must be MACROMOLECULE');\n        const stats = WebLogo.getStats(col, 5, WebLogo.splitterAsFasta);\n        const seqType = stats.sameLength ? 'SEQ.MSA' : 'SEQ';\n        const alphabetCandidates = [\n            [\"PT\" /* ALPHABET.PT */, UnitsHandler.PeptideFastaAlphabet],\n            [\"DNA\" /* ALPHABET.DNA */, UnitsHandler.DnaFastaAlphabet],\n            [\"RNA\" /* ALPHABET.RNA */, UnitsHandler.RnaFastaAlphabet],\n        ];\n        // Calculate likelihoods for alphabet_candidates\n        const alphabetCandidatesSim = alphabetCandidates.map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[1]));\n        const maxCos = Math.max(...alphabetCandidatesSim);\n        const alphabet = maxCos > 0.65 ? alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][0] : 'UN';\n        col.setTag(DG.TAGS.UNITS, \"fasta\" /* NOTATION.FASTA */);\n        col.setTag(UnitsHandler.TAGS.aligned, seqType);\n        col.setTag(UnitsHandler.TAGS.alphabet, alphabet);\n    }\n    get units() { return this._units; }\n    get column() { return this._column; }\n    get notation() { return this._notation; }\n    get defaultGapSymbol() { return this._defaultGapSymbol; }\n    get separator() {\n        const separator = this.column.getTag(UnitsHandler.TAGS.separator);\n        if (separator !== null)\n            return separator;\n        else\n            throw new Error('Separator not set');\n    }\n    get aligned() {\n        const aligned = this.column.getTag(UnitsHandler.TAGS.aligned);\n        if (aligned !== null) {\n            return aligned;\n        }\n        else {\n            throw new Error('Tag aligned not set');\n        }\n    }\n    /** Alphabet name (upper case) */\n    get alphabet() {\n        const alphabet = this.column.getTag(UnitsHandler.TAGS.alphabet);\n        if (alphabet !== null) {\n            return alphabet.toUpperCase();\n        }\n        else {\n            throw new Error('Tag alphabet not set');\n        }\n    }\n    getAlphabetSize() {\n        if (this.notation == \"helm\" /* NOTATION.HELM */ || this.alphabet == \"UN\" /* ALPHABET.UN */) {\n            const alphabetSize = parseInt(this.column.getTag(UnitsHandler.TAGS.alphabetSize));\n            return alphabetSize;\n        }\n        else {\n            switch (this.alphabet) {\n                case \"PT\" /* ALPHABET.PT */:\n                    return 20;\n                case \"DNA\" /* ALPHABET.DNA */:\n                case \"RNA\" /* ALPHABET.RNA */:\n                    return 4;\n                case 'NT':\n                    console.warn(`Unexpected alphabet 'NT'.`);\n                    return 4;\n                default:\n                    throw new Error(`Unexpected alphabet '${this.alphabet}'.`);\n            }\n        }\n    }\n    getAlphabetIsMultichar() {\n        if (this.notation == \"helm\" /* NOTATION.HELM */ || this.alphabet == \"UN\" /* ALPHABET.UN */) {\n            return this.column.getTag(UnitsHandler.TAGS.alphabetIsMultichar) == 'true';\n        }\n        else {\n            return false;\n        }\n    }\n    isFasta() { return this.notation === \"fasta\" /* NOTATION.FASTA */; }\n    isSeparator() { return this.notation === \"separator\" /* NOTATION.SEPARATOR */; }\n    isHelm() { return this.notation === \"helm\" /* NOTATION.HELM */; }\n    isRna() { return this.alphabet === 'RNA'; }\n    isDna() { return this.alphabet === 'DNA'; }\n    isPeptide() { return this.alphabet === 'PT'; }\n    /** Associate notation types with the corresponding units */\n    /**\n     * @return {NOTATION}     Notation associated with the units type\n     */\n    getNotation() {\n        if (this.units.toLowerCase().startsWith(\"fasta\" /* NOTATION.FASTA */))\n            return \"fasta\" /* NOTATION.FASTA */;\n        else if (this.units.toLowerCase().startsWith(\"separator\" /* NOTATION.SEPARATOR */))\n            return \"separator\" /* NOTATION.SEPARATOR */;\n        else if (this.units.toLowerCase().startsWith(\"helm\" /* NOTATION.HELM */))\n            return \"helm\" /* NOTATION.HELM */;\n        else\n            throw new Error(`Column '${this.column.name}' has unexpected notation '${this.units}'.`);\n    }\n    /**\n     * Create a new empty column of the specified notation type and the same\n     * length as column\n     *\n     * @param {NOTATION} targetNotation\n     * @return {DG.Column}\n     */\n    getNewColumn(targetNotation) {\n        const col = this.column;\n        const len = col.length;\n        const name = targetNotation.toLowerCase() + '(' + col.name + ')';\n        const newColName = col.dataFrame.columns.getUnusedName(name);\n        const newColumn = DG.Column.fromList('string', newColName, new Array(len).fill(''));\n        newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n        newColumn.setTag(DG.TAGS.UNITS, this.notation);\n        newColumn.setTag(DG.TAGS.CELL_RENDERER, 'Macromolecule');\n        const srcAlphabet = col.getTag(UnitsHandler.TAGS.alphabet);\n        if (srcAlphabet)\n            newColumn.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n        const srcAlphabetSize = col.getTag(UnitsHandler.TAGS.alphabetSize);\n        if (srcAlphabetSize)\n            newColumn.setTag(UnitsHandler.TAGS.alphabetSize, srcAlphabetSize);\n        const srcAlphabetIsMultichar = col.getTag(UnitsHandler.TAGS.alphabetIsMultichar);\n        if (srcAlphabetIsMultichar)\n            newColumn.setTag(UnitsHandler.TAGS.alphabetIsMultichar, srcAlphabetIsMultichar);\n        return newColumn;\n    }\n    /**\n     * Create a new empty column using templateCol as a template\n     *\n     * @param {DG.Column} templateCol  the properties and units of this column are used as a\n     * template to build the new one\n     * @return {DG.Column}\n     */\n    static getNewColumn(templateCol) {\n        const col = new UnitsHandler(templateCol);\n        const targetNotation = col.notation;\n        return col.getNewColumn(targetNotation);\n    }\n    /**\n     * A helper function checking the validity of the 'units' string\n     *\n     * @param {string} units  the string to be validated\n     * @return {boolean}\n     */\n    static unitsStringIsValid(units) {\n        units = units.toLowerCase();\n        const prefixes = [\"fasta\" /* NOTATION.FASTA */, \"separator\" /* NOTATION.SEPARATOR */, \"helm\" /* NOTATION.HELM */];\n        const postfixes = ['rna', 'dna', 'pt'];\n        const prefixCriterion = prefixes.some((p) => units.startsWith(p.toLowerCase()));\n        return prefixCriterion;\n    }\n    /**\n     * Construct a new column of semantic type MACROMOLECULE from the list of\n     * specified parameters\n     *\n     * @param {number}    len  the length of the new column\n     * @param {string}    name  the name of the new column\n     * @param {string}    units  the units of the new column\n     * @return {DG.Column}\n     */\n    static getNewColumnFromParams(len, name, units) {\n        // WARNING: in this implementation is is impossible to verify the uniqueness\n        // of the new column's name\n        // TODO: verify the validity of units parameter\n        if (!UnitsHandler.unitsStringIsValid(units))\n            throw new Error('Invalid format of \\'units\\' parameter');\n        const newColumn = DG.Column.fromList('string', name, new Array(len).fill(''));\n        newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n        newColumn.setTag(DG.TAGS.UNITS, units);\n        return newColumn;\n    }\n}\nUnitsHandler.TAGS = {\n    aligned: 'aligned',\n    alphabet: 'alphabet',\n    alphabetSize: '.alphabetSize',\n    alphabetIsMultichar: '.alphabetIsMultichar',\n    separator: 'separator',\n};\nUnitsHandler._defaultGapSymbolsDict = {\n    HELM: '*',\n    SEPARATOR: '',\n    FASTA: '-',\n};\nUnitsHandler.PeptideFastaAlphabet = new Set([\n    'G', 'L', 'Y', 'S', 'E', 'Q', 'D', 'N', 'F', 'A',\n    'K', 'R', 'H', 'C', 'V', 'P', 'W', 'I', 'M', 'T',\n]);\nUnitsHandler.DnaFastaAlphabet = new Set(['A', 'C', 'G', 'T']);\nUnitsHandler.RnaFastaAlphabet = new Set(['A', 'C', 'G', 'U']);\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"units-handler.js","sourceRoot":"","sources":["units-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport * as grok from 'datagrok-api/grok';\n\nimport {WebLogo, SeqColStats} from '../viewers/web-logo';\n\n/** enum type to simplify setting \"user-friendly\" notation if necessary */\nexport const enum NOTATION {\n  FASTA = 'fasta',\n  SEPARATOR = 'separator',\n  HELM = 'helm',\n}\n\nexport const enum ALPHABET {\n  DNA = 'DNA',\n  RNA = 'RNA',\n  PT = 'PT',\n  UN = 'UN',\n}\n\n/** Class for handling notation units in Macromolecule columns */\nexport class UnitsHandler {\n  public static readonly TAGS = {\n    aligned: 'aligned',\n    alphabet: 'alphabet',\n    alphabetSize: '.alphabetSize',\n    alphabetIsMultichar: '.alphabetIsMultichar',\n    separator: 'separator',\n  };\n\n  protected readonly _column: DG.Column; // the column to be converted\n  protected _units: string; // units, of the form fasta, separator\n  protected _notation: NOTATION; // current notation (without :SEQ:NT, etc.)\n  protected _defaultGapSymbol: string;\n  protected static readonly _defaultGapSymbolsDict = {\n    HELM: '*',\n    SEPARATOR: '',\n    FASTA: '-',\n  };\n\n  public static readonly PeptideFastaAlphabet = new Set([\n    'G', 'L', 'Y', 'S', 'E', 'Q', 'D', 'N', 'F', 'A',\n    'K', 'R', 'H', 'C', 'V', 'P', 'W', 'I', 'M', 'T',\n  ]);\n  public static readonly DnaFastaAlphabet = new Set(['A', 'C', 'G', 'T']);\n  public static readonly RnaFastaAlphabet = new Set(['A', 'C', 'G', 'U']);\n\n  public static setUnitsToFastaColumn(col: DG.Column) {\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n      throw new Error('Fasta column must be MACROMOLECULE');\n\n    const stats: SeqColStats = WebLogo.getStats(col, 5, WebLogo.splitterAsFasta);\n    const seqType = stats.sameLength ? 'SEQ.MSA' : 'SEQ';\n\n    const alphabetCandidates: [string, Set<string>][] = [\n      [ALPHABET.PT, UnitsHandler.PeptideFastaAlphabet],\n      [ALPHABET.DNA, UnitsHandler.DnaFastaAlphabet],\n      [ALPHABET.RNA, UnitsHandler.RnaFastaAlphabet],\n    ];\n\n    // Calculate likelihoods for alphabet_candidates\n    const alphabetCandidatesSim: number[] = alphabetCandidates.map(\n      (c) => WebLogo.getAlphabetSimilarity(stats.freq, c[1]));\n    const maxCos = Math.max(...alphabetCandidatesSim);\n    const alphabet = maxCos > 0.65 ? alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][0] : 'UN';\n\n    col.setTag(DG.TAGS.UNITS, NOTATION.FASTA);\n    col.setTag(UnitsHandler.TAGS.aligned, seqType);\n    col.setTag(UnitsHandler.TAGS.alphabet, alphabet);\n  }\n\n  protected get units(): string { return this._units; }\n\n  protected get column(): DG.Column { return this._column; }\n\n  public get notation(): NOTATION { return this._notation; }\n\n  public get defaultGapSymbol(): string { return this._defaultGapSymbol; }\n\n  public get separator(): string {\n    const separator = this.column.getTag(UnitsHandler.TAGS.separator);\n    if (separator !== null)\n      return separator;\n    else\n      throw new Error('Separator not set');\n  }\n\n  public get aligned(): string {\n    const aligned = this.column.getTag(UnitsHandler.TAGS.aligned);\n    if (aligned !== null) {\n      return aligned;\n    } else {\n      throw new Error('Tag aligned not set');\n    }\n  }\n\n  /** Alphabet name (upper case) */\n  public get alphabet(): string {\n    const alphabet = this.column.getTag(UnitsHandler.TAGS.alphabet);\n    if (alphabet !== null) {\n      return alphabet.toUpperCase();\n    } else {\n      throw new Error('Tag alphabet not set');\n    }\n  }\n\n  public getAlphabetSize(): number {\n    if (this.notation == NOTATION.HELM || this.alphabet == ALPHABET.UN) {\n      const alphabetSize = parseInt(this.column.getTag(UnitsHandler.TAGS.alphabetSize));\n      return alphabetSize;\n    } else {\n      switch (this.alphabet) {\n      case ALPHABET.PT:\n        return 20;\n      case ALPHABET.DNA:\n      case ALPHABET.RNA:\n        return 4;\n      case 'NT':\n        console.warn(`Unexpected alphabet 'NT'.`);\n        return 4;\n      default:\n        throw new Error(`Unexpected alphabet '${this.alphabet}'.`);\n      }\n    }\n  }\n\n  public getAlphabetIsMultichar(): boolean {\n    if (this.notation == NOTATION.HELM || this.alphabet == ALPHABET.UN) {\n      return this.column.getTag(UnitsHandler.TAGS.alphabetIsMultichar) == 'true';\n    } else {\n      return false;\n    }\n  }\n\n  public isFasta(): boolean { return this.notation === NOTATION.FASTA; }\n\n  public isSeparator(): boolean { return this.notation === NOTATION.SEPARATOR; }\n\n  public isHelm(): boolean { return this.notation === NOTATION.HELM; }\n\n  public isRna(): boolean { return this.alphabet === 'RNA'; }\n\n  public isDna(): boolean { return this.alphabet === 'DNA'; }\n\n  public isPeptide(): boolean { return this.alphabet === 'PT'; }\n\n  /** Associate notation types with the corresponding units */\n  /**\n   * @return {NOTATION}     Notation associated with the units type\n   */\n  protected getNotation(): NOTATION {\n    if (this.units.toLowerCase().startsWith(NOTATION.FASTA))\n      return NOTATION.FASTA;\n    else if (this.units.toLowerCase().startsWith(NOTATION.SEPARATOR))\n      return NOTATION.SEPARATOR;\n    else if (this.units.toLowerCase().startsWith(NOTATION.HELM))\n      return NOTATION.HELM;\n    else\n      throw new Error(`Column '${this.column.name}' has unexpected notation '${this.units}'.`);\n  }\n\n  /**\n   * Create a new empty column of the specified notation type and the same\n   * length as column\n   *\n   * @param {NOTATION} targetNotation\n   * @return {DG.Column}\n   */\n  protected getNewColumn(targetNotation: NOTATION): DG.Column {\n    const col = this.column;\n    const len = col.length;\n    const name = targetNotation.toLowerCase() + '(' + col.name + ')';\n    const newColName = col.dataFrame.columns.getUnusedName(name);\n    const newColumn = DG.Column.fromList('string', newColName, new Array(len).fill(''));\n    newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n    newColumn.setTag(DG.TAGS.UNITS, this.notation);\n    newColumn.setTag(DG.TAGS.CELL_RENDERER, 'Macromolecule');\n\n    const srcAlphabet = col.getTag(UnitsHandler.TAGS.alphabet);\n    if (srcAlphabet)\n      newColumn.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n\n    const srcAlphabetSize = col.getTag(UnitsHandler.TAGS.alphabetSize);\n    if (srcAlphabetSize)\n      newColumn.setTag(UnitsHandler.TAGS.alphabetSize, srcAlphabetSize);\n\n    const srcAlphabetIsMultichar = col.getTag(UnitsHandler.TAGS.alphabetIsMultichar);\n    if (srcAlphabetIsMultichar)\n      newColumn.setTag(UnitsHandler.TAGS.alphabetIsMultichar, srcAlphabetIsMultichar);\n\n    return newColumn;\n  }\n\n  /**\n   * Create a new empty column using templateCol as a template\n   *\n   * @param {DG.Column} templateCol  the properties and units of this column are used as a\n   * template to build the new one\n   * @return {DG.Column}\n   */\n  public static getNewColumn(templateCol: DG.Column): DG.Column {\n    const col: UnitsHandler = new UnitsHandler(templateCol);\n    const targetNotation = col.notation;\n    return col.getNewColumn(targetNotation);\n  }\n\n  /**\n   * A helper function checking the validity of the 'units' string\n   *\n   * @param {string} units  the string to be validated\n   * @return {boolean}\n   */\n  public static unitsStringIsValid(units: string): boolean {\n    units = units.toLowerCase();\n    const prefixes = [NOTATION.FASTA, NOTATION.SEPARATOR, NOTATION.HELM];\n    const postfixes = ['rna', 'dna', 'pt'];\n\n    const prefixCriterion = prefixes.some((p) => units.startsWith(p.toLowerCase()));\n    return prefixCriterion;\n  }\n\n  /**\n   * Construct a new column of semantic type MACROMOLECULE from the list of\n   * specified parameters\n   *\n   * @param {number}    len  the length of the new column\n   * @param {string}    name  the name of the new column\n   * @param {string}    units  the units of the new column\n   * @return {DG.Column}\n   */\n  public static getNewColumnFromParams(\n    len: number,\n    name: string,\n    units: string\n  ): DG.Column {\n    // WARNING: in this implementation is is impossible to verify the uniqueness\n    // of the new column's name\n    // TODO: verify the validity of units parameter\n    if (!UnitsHandler.unitsStringIsValid(units))\n      throw new Error('Invalid format of \\'units\\' parameter');\n    const newColumn = DG.Column.fromList('string', name, new Array(len).fill(''));\n    newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n    newColumn.setTag(DG.TAGS.UNITS, units);\n    return newColumn;\n  }\n\n  public constructor(col: DG.Column) {\n    this._column = col;\n    const units = this._column.tags[DG.TAGS.UNITS];\n    if (units !== null)\n      this._units = units;\n    else\n      throw new Error('Units are not specified in column');\n    this._notation = this.getNotation();\n    this._defaultGapSymbol = (this.isFasta()) ? UnitsHandler._defaultGapSymbolsDict.FASTA :\n      (this.isHelm()) ? UnitsHandler._defaultGapSymbolsDict.HELM :\n        UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n\n    if (!this.column.tags.has(UnitsHandler.TAGS.alphabetSize)) {\n      if (this.isHelm())\n        throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n      else if (['UN'].includes(this.alphabet))\n        throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n    }\n\n    if (!this.column.tags.has(UnitsHandler.TAGS.alphabetIsMultichar)) {\n      if (this.isHelm())\n        throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n      else if (['UN'].includes(this.alphabet))\n        throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n    }\n  }\n}\n"]}","// Data structures for V-Domain regions of antibodies\nexport var VdRegionType;\n(function (VdRegionType) {\n    VdRegionType[\"Unknown\"] = \"unknown\";\n    VdRegionType[\"FR\"] = \"framework\";\n    VdRegionType[\"CDR\"] = \"cdr\";\n})(VdRegionType || (VdRegionType = {}));\n/** Describes V-DOMAIN (IG and TR) region (of multiple alignment)\n * https://www.imgt.org/IMGTScientificChart/Numbering/IMGTIGVLsuperfamily.html\n */\nexport class VdRegion {\n    /**\n     * start and position are strings because they correspond to position names as column names in ANARCI output\n     * @param {VdRegionType} type  Type of the region\n     * @param {string} name  Name of the region\n     * @param {string} chain\n     * @param {string} order\n     * @param {string} positionStartName  Region start position (inclusive)\n     * @param {string} positionEndName  Region end position (inclusive)\n     */\n    constructor(type, name, chain, order, positionStartName, positionEndName) {\n        this.type = type;\n        this.name = name;\n        this.chain = chain;\n        this.order = order;\n        this.positionStartName = positionStartName;\n        this.positionEndName = positionEndName;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmQtcmVnaW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInZkLXJlZ2lvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEscURBQXFEO0FBRXJELE1BQU0sQ0FBTixJQUFZLFlBSVg7QUFKRCxXQUFZLFlBQVk7SUFDdEIsbUNBQW1CLENBQUE7SUFDbkIsZ0NBQWdCLENBQUE7SUFDaEIsMkJBQVcsQ0FBQTtBQUNiLENBQUMsRUFKVyxZQUFZLEtBQVosWUFBWSxRQUl2QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFFBQVE7SUFRbkI7Ozs7Ozs7O09BUUc7SUFDSCxZQUFZLElBQWtCLEVBQUUsSUFBWSxFQUFFLEtBQWEsRUFBRSxLQUFhLEVBQ3hFLGlCQUF5QixFQUFFLGVBQXVCO1FBQ2xELElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxpQkFBaUIsR0FBRyxpQkFBaUIsQ0FBQztRQUMzQyxJQUFJLENBQUMsZUFBZSxHQUFHLGVBQWUsQ0FBQztJQUN6QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBEYXRhIHN0cnVjdHVyZXMgZm9yIFYtRG9tYWluIHJlZ2lvbnMgb2YgYW50aWJvZGllc1xuXG5leHBvcnQgZW51bSBWZFJlZ2lvblR5cGUge1xuICBVbmtub3duID0gJ3Vua25vd24nLFxuICBGUiA9ICdmcmFtZXdvcmsnLFxuICBDRFIgPSAnY2RyJyxcbn1cblxuLyoqIERlc2NyaWJlcyBWLURPTUFJTiAoSUcgYW5kIFRSKSByZWdpb24gKG9mIG11bHRpcGxlIGFsaWdubWVudClcbiAqIGh0dHBzOi8vd3d3LmltZ3Qub3JnL0lNR1RTY2llbnRpZmljQ2hhcnQvTnVtYmVyaW5nL0lNR1RJR1ZMc3VwZXJmYW1pbHkuaHRtbFxuICovXG5leHBvcnQgY2xhc3MgVmRSZWdpb24ge1xuICB0eXBlOiBWZFJlZ2lvblR5cGU7XG4gIG5hbWU6IHN0cmluZztcbiAgY2hhaW46IHN0cmluZztcbiAgb3JkZXI6IG51bWJlcjtcbiAgcG9zaXRpb25TdGFydE5hbWU6IHN0cmluZztcbiAgcG9zaXRpb25FbmROYW1lOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIHN0YXJ0IGFuZCBwb3NpdGlvbiBhcmUgc3RyaW5ncyBiZWNhdXNlIHRoZXkgY29ycmVzcG9uZCB0byBwb3NpdGlvbiBuYW1lcyBhcyBjb2x1bW4gbmFtZXMgaW4gQU5BUkNJIG91dHB1dFxuICAgKiBAcGFyYW0ge1ZkUmVnaW9uVHlwZX0gdHlwZSAgVHlwZSBvZiB0aGUgcmVnaW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBuYW1lICBOYW1lIG9mIHRoZSByZWdpb25cbiAgICogQHBhcmFtIHtzdHJpbmd9IGNoYWluXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBvcmRlclxuICAgKiBAcGFyYW0ge3N0cmluZ30gcG9zaXRpb25TdGFydE5hbWUgIFJlZ2lvbiBzdGFydCBwb3NpdGlvbiAoaW5jbHVzaXZlKVxuICAgKiBAcGFyYW0ge3N0cmluZ30gcG9zaXRpb25FbmROYW1lICBSZWdpb24gZW5kIHBvc2l0aW9uIChpbmNsdXNpdmUpXG4gICAqL1xuICBjb25zdHJ1Y3Rvcih0eXBlOiBWZFJlZ2lvblR5cGUsIG5hbWU6IHN0cmluZywgY2hhaW46IHN0cmluZywgb3JkZXI6IG51bWJlcixcbiAgICBwb3NpdGlvblN0YXJ0TmFtZTogc3RyaW5nLCBwb3NpdGlvbkVuZE5hbWU6IHN0cmluZykge1xuICAgIHRoaXMudHlwZSA9IHR5cGU7XG4gICAgdGhpcy5uYW1lID0gbmFtZTtcbiAgICB0aGlzLmNoYWluID0gY2hhaW47XG4gICAgdGhpcy5vcmRlciA9IG9yZGVyO1xuICAgIHRoaXMucG9zaXRpb25TdGFydE5hbWUgPSBwb3NpdGlvblN0YXJ0TmFtZTtcbiAgICB0aGlzLnBvc2l0aW9uRW5kTmFtZSA9IHBvc2l0aW9uRW5kTmFtZTtcbiAgfVxufVxuIl19","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport wu from 'wu';\nimport * as rxjs from 'rxjs';\nimport { Vector } from '@datagrok-libraries/utils/src/type-declarations';\nimport { vectorLength, vectorDotProduct } from '@datagrok-libraries/utils/src/vector-operations';\nimport { Aminoacids, AminoacidsPalettes } from '../aminoacids';\nimport { Nucleotides, NucleotidesPalettes } from '../nucleotides';\nimport { UnknownSeqPalettes } from '../unknown';\nimport { UnitsHandler } from '../utils/units-handler';\nvar PositionHeight;\n(function (PositionHeight) {\n    PositionHeight[\"Entropy\"] = \"Entropy\";\n    PositionHeight[\"full\"] = \"100%\";\n})(PositionHeight || (PositionHeight = {}));\n/**@param {MouseEvent} event\n * @param {number} r devicePixelRation\n * @return {DG.Point} canvas related cursor position\n */\nHTMLCanvasElement.prototype.getCursorPosition = function (event, r) {\n    const rect = this.getBoundingClientRect();\n    return new DG.Point((event.clientX - rect.left) * r, (event.clientY - rect.top) * r);\n};\nDG.Rect.prototype.contains = function (x, y) {\n    return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;\n};\nexport class PositionMonomerInfo {\n    constructor(count = 0, bounds = new DG.Rect(0, 0, 0, 0)) {\n        this.count = count;\n        this.bounds = bounds;\n    }\n}\nexport class PositionInfo {\n    /** freq = {}, rowCount = 0\n     * @param {string} name Name of position ('111A', '111.1', etc)\n     * @param {number} sumForHeightCalc Sum of all monomer counts for height calculation\n     * @param {number} rowCount Count of elements in column\n     * @param {string[]} freq frequency of monomers in position\n     */\n    constructor(name, freq = {}, rowCount = 0, sumForHeightCalc = 0) {\n        this.name = name;\n        this.freq = freq;\n        this.rowCount = rowCount;\n        this.sumForHeightCalc = sumForHeightCalc;\n    }\n}\nexport class WebLogo extends DG.JsViewer {\n    constructor() {\n        super();\n        this.viewerId = -1;\n        this.initialized = false;\n        // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];\n        this.cp = null;\n        this.axisHeight = 12;\n        this.seqCol = null;\n        this.splitter = null;\n        // private maxLength: number = 100;\n        this.positions = [];\n        this.rowsMasked = 0;\n        this.rowsNull = 0;\n        this.positionMargin = 0;\n        this.positionNames = [];\n        this.startPosition = -1;\n        this.endPosition = -1;\n        this.viewSubs = [];\n        this.viewerId = WebLogo.viewerCount;\n        WebLogo.viewerCount += 1;\n        this.textBaseline = 'top';\n        this.unitsHandler = null;\n        this._positionWidth = this.positionWidth = this.float('positionWidth', 16 /*,\n          {editor: 'slider', min: 4, max: 64, postfix: 'px'}*/);\n        this.minHeight = this.float('minHeight', 50 /*,\n          {editor: 'slider', min: 25, max: 250, postfix: 'px'}*/);\n        this.maxHeight = this.float('maxHeight', 100 /*,\n          {editor: 'slider', min: 25, max: 500, postfix: 'px'}*/);\n        this.considerNullSequences = this.bool('considerNullSequences', false);\n        this.sequenceColumnName = this.string('sequenceColumnName', null);\n        this.startPositionName = this.string('startPositionName', null);\n        this.endPositionName = this.string('endPositionName', null);\n        this.fixWidth = this.bool('fixWidth', false);\n        this.verticalAlignment = this.string('verticalAlignment', 'middle', { choices: ['top', 'middle', 'bottom'] });\n        this.horizontalAlignment = this.string('horizontalAlignment', 'center', { choices: ['left', 'center', 'right'] });\n        this.fitArea = this.bool('fitArea', true);\n        this.shrinkEmptyTail = this.bool('shrinkEmptyTail', true);\n        this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n        this.positionMarginState = this.string('positionMarginState', 'auto', { choices: ['auto', 'enable', 'off'] });\n        let defaultValueForPositionMargin = 0;\n        if (this.positionMarginState === 'auto') {\n            defaultValueForPositionMargin = 4;\n        }\n        this.positionMargin = this.int('positionMargin', defaultValueForPositionMargin, { min: 0, max: 16 });\n        this.positionHeight = this.string('positionHeight', PositionHeight.full, { choices: [PositionHeight.full, PositionHeight.Entropy] });\n    }\n    /** For startPosition equals to endPosition Length is 1 */\n    get Length() {\n        if (this.skipEmptyPositions) {\n            return this.positions.length;\n        }\n        return this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0;\n    }\n    get positionWidthWithMargin() {\n        var _a;\n        if ((this.positionMarginState === 'auto') && (((_a = this.unitsHandler) === null || _a === void 0 ? void 0 : _a.getAlphabetIsMultichar()) === true)) {\n            return this._positionWidth + this.positionMargin;\n        }\n        if (this.positionMarginState === 'enable') {\n            return this._positionWidth + this.positionMargin;\n        }\n        return this._positionWidth;\n    }\n    init() {\n        return __awaiter(this, void 0, void 0, function* () {\n            if (this.initialized) {\n                console.error('WebLogo second initialization!');\n                return;\n            }\n            this.initialized = true;\n            this.helpUrl = '/help/visualize/viewers/web-logo.md';\n            this.msgHost = ui.div('No message');\n            this.msgHost.style.display = 'none';\n            this.canvas = ui.canvas();\n            this.canvas.style.width = '100%';\n            this.host = ui.div([this.msgHost, this.canvas]);\n            // this.slider = ui.rangeSlider(0, 20, 2, 5);\n            // this.slider.root.style.width = '100%';\n            // this.slider.root.style.height = '12px';\n            const getMonomer = (p) => {\n                const jPos = Math.floor(p.x / this.positionWidthWithMargin);\n                const position = this.positions[jPos];\n                if (position === void 0)\n                    return [jPos, null, null];\n                const monomer = Object.keys(position.freq)\n                    .find((m) => position.freq[m].bounds.contains(p.x, p.y));\n                if (monomer === undefined)\n                    return [jPos, null, null];\n                return [jPos, monomer, position.freq[monomer]];\n            };\n            // this.subs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mouseover').subscribe((e: MouseEvent) => {\n            // }));\n            this.viewSubs.push(rxjs.fromEvent(this.canvas, 'mousemove').subscribe((e) => {\n                if (!this.canvas)\n                    return;\n                const args = e;\n                const r = window.devicePixelRatio;\n                const cursorP = this.canvas.getCursorPosition(args, r);\n                const [jPos, monomer] = getMonomer(cursorP);\n                if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n                    const rowCount = wu.count().take(this.dataFrame.rowCount).filter((iRow) => {\n                        const seq = this.seqCol.get(iRow);\n                        const seqM = seq ? this.splitter(seq)[this.startPosition + jPos] : null;\n                        return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n                    }).reduce((count, iRow) => count + 1, 0);\n                    ui.tooltip.show(ui.div([ui.div(`${monomer}`), ui.div(`${rowCount} rows`)]), args.x + 16, args.y + 16);\n                }\n                else {\n                    ui.tooltip.hide();\n                }\n            }));\n            this.viewSubs.push(rxjs.fromEvent(this.canvas, 'mousedown').subscribe((e) => {\n                if (!this.canvas || e.button != 0)\n                    return;\n                const args = e;\n                const r = window.devicePixelRatio;\n                const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args, r));\n                // prevents deselect all rows if we miss monomer bounds\n                if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n                    this.dataFrame.selection.init((iRow) => {\n                        const seq = this.seqCol.get(iRow);\n                        const seqM = seq ? this.splitter(seq)[this.startPosition + jPos] : null;\n                        return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n                    });\n                }\n            }));\n            this.viewSubs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n            this.root.append(this.host);\n            // this.root.appendChild(this.slider.root);\n            this.render(true);\n        });\n    }\n    rootOnSizeChanged(args) {\n        this.render(true);\n        // console.debug(`WebLogo.onRootSizeChanged() ` +\n        //   `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +\n        //   `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);\n    }\n    /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().\n     */\n    updateSeqCol() {\n        if (this.dataFrame) {\n            this.seqCol = this.sequenceColumnName ? this.dataFrame.col(this.sequenceColumnName) : null;\n            if (this.seqCol == null) {\n                this.seqCol = WebLogo.pickUpSeqCol2(this.dataFrame);\n                this.sequenceColumnName = this.seqCol ? this.seqCol.name : null;\n            }\n            if (this.seqCol) {\n                const units = this.seqCol.getTag(DG.TAGS.UNITS);\n                const separator = this.seqCol.getTag(UnitsHandler.TAGS.separator);\n                this.splitter = WebLogo.getSplitter(units, separator);\n                this.unitsHandler = new UnitsHandler(this.seqCol);\n                this.updatePositions();\n                this.cp = WebLogo.pickUpPalette(this.seqCol);\n            }\n            else {\n                this.splitter = null;\n                this.positionNames = [];\n                this.startPosition = -1;\n                this.endPosition = -1;\n                this.cp = null;\n            }\n        }\n        this.render();\n    }\n    updatePositions() {\n        if (!this.seqCol)\n            return;\n        let categories;\n        if (this.shrinkEmptyTail) {\n            const indices = this.dataFrame.filter.getSelectedIndexes();\n            categories = Array.from(new Set(Array.from(Array(indices.length).keys()).map((i) => this.seqCol.get(indices[i]))));\n        }\n        else {\n            categories = this.seqCol.categories;\n        }\n        const maxLength = categories.length > 0 ? Math.max(...categories.map((s) => s !== null ? this.splitter(s).length : 0)) : 0;\n        // Get position names from data column tag 'positionNames'\n        const positionNamesTxt = this.seqCol.getTag('positionNames');\n        // Fallback if 'positionNames' tag is not provided\n        this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :\n            [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);\n        this.startPosition = (this.startPositionName && this.positionNames &&\n            this.positionNames.includes(this.startPositionName)) ?\n            this.positionNames.indexOf(this.startPositionName) : 0;\n        this.endPosition = (this.endPositionName && this.positionNames &&\n            this.positionNames.includes(this.endPositionName)) ?\n            this.positionNames.indexOf(this.endPositionName) : (maxLength - 1);\n    }\n    onPropertyChanged(property) {\n        // console.debug(`WebLogo.onPropertyChanged( ${property.name} = '' })`);\n        super.onPropertyChanged(property);\n        switch (property.name) {\n            case 'considerNullSequences':\n                this.render(true);\n                break;\n            case 'sequenceColumnName':\n                this.updateSeqCol();\n                break;\n            case 'startPositionName':\n                this.updateSeqCol();\n                break;\n            case 'endPositionName':\n                this.updateSeqCol();\n                break;\n            case 'positionWidth':\n                this._positionWidth = this.positionWidth;\n                this.render(true);\n                break;\n            case 'minHeight':\n                this.render(true);\n                break;\n            case 'maxHeight':\n                this.render(true);\n                break;\n            case 'fixWidth':\n                this.render(true);\n                break;\n            case 'verticalAlignment':\n                this.render(true);\n                break;\n            case 'horizontalAlignment':\n                this.render(true);\n                break;\n            case 'fitArea':\n                this.render(true);\n                break;\n            case 'shrinkEmptyTail':\n                this.updatePositions();\n                this.render(true);\n                break;\n            case 'skipEmptyPositions':\n                this.updatePositions();\n                this.render(true);\n                break;\n            case 'positionMargin':\n                this.render(true);\n                break;\n            case 'positionMarginState':\n                this.render(true);\n                break;\n            case 'positionHeight':\n                this.render(true);\n                break;\n        }\n    }\n    onTableAttached() {\n        const _super = Object.create(null, {\n            onTableAttached: { get: () => super.onTableAttached }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            const dataFrameTxt = this.dataFrame ? 'data' : 'null';\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n            _super.onTableAttached.call(this);\n            this.updateSeqCol();\n            if (this.dataFrame !== void 0) {\n                this.subs.push(this.dataFrame.selection.onChanged.subscribe((_) => this.render()));\n                this.subs.push(this.dataFrame.filter.onChanged.subscribe((_) => {\n                    this.updatePositions();\n                    this.render();\n                }));\n            }\n            yield this.init();\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n        });\n    }\n    detach() {\n        const _super = Object.create(null, {\n            detach: { get: () => super.detach }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            const dataFrameTxt = `${this.dataFrame ? 'data' : 'null'}`;\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n            _super.detach.call(this);\n            this.viewSubs.forEach((sub) => sub.unsubscribe());\n            this.host.remove();\n            this.canvas = undefined;\n            this.msgHost = undefined;\n            this.host = undefined;\n            this.initialized = false;\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n        });\n    }\n    _nullSequence(fillerResidue = 'X') {\n        if (this.considerNullSequences)\n            return new Array(this.Length).fill(fillerResidue).join('');\n        return '';\n    }\n    removeWhere(array, predicate) {\n        let length = array.length;\n        let updateIterator = 0;\n        for (let deleteIterator = 0; deleteIterator < length; deleteIterator++) {\n            if (!predicate(array[deleteIterator])) {\n                array[updateIterator] = array[deleteIterator];\n                updateIterator++;\n            }\n        }\n        array.length = updateIterator;\n        return array;\n    }\n    _removeEmptyPositions() {\n        if (this.skipEmptyPositions) {\n            this.removeWhere(this.positions, item => { var _a; return ((_a = item === null || item === void 0 ? void 0 : item.freq['-']) === null || _a === void 0 ? void 0 : _a.count) === item.rowCount; });\n        }\n    }\n    _calculate(r) {\n        if (!this.canvas || !this.host || !this.seqCol || !this.dataFrame)\n            return;\n        this.unitsHandler = new UnitsHandler(this.seqCol);\n        this.calcSize();\n        this.positions = new Array(this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0);\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const posName = this.positionNames[this.startPosition + jPos];\n            this.positions[jPos] = new PositionInfo(posName);\n        }\n        // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)\n        const indices = this.dataFrame.filter.getSelectedIndexes();\n        // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :\n        //   this.dataFrame.filter.getSelectedIndexes();\n        this.rowsMasked = indices.length;\n        this.rowsNull = 0;\n        for (const i of indices) {\n            let s = (this.seqCol.get(i));\n            if (!s) {\n                s = this._nullSequence();\n                ++this.rowsNull;\n            }\n            const seqM = this.splitter(s);\n            for (let jPos = 0; jPos < this.Length; jPos++) {\n                const pmInfo = this.positions[jPos].freq;\n                const m = seqM[this.startPosition + jPos] || '-';\n                if (!(m in pmInfo))\n                    pmInfo[m] = new PositionMonomerInfo();\n                pmInfo[m].count++;\n            }\n        }\n        //#region Polish freq counts\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            // delete this.positions[jPos].freq['-'];\n            this.positions[jPos].rowCount = 0;\n            for (const m in this.positions[jPos].freq)\n                this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;\n            if (this.positionHeight == PositionHeight.Entropy) {\n                this.positions[jPos].sumForHeightCalc = 0;\n                for (const m in this.positions[jPos].freq) {\n                    const pn = this.positions[jPos].freq[m].count / this.positions[jPos].rowCount;\n                    this.positions[jPos].sumForHeightCalc += -pn * Math.log2(pn);\n                }\n            }\n        }\n        //#endregion\n        this._removeEmptyPositions();\n        const absoluteMaxHeight = this.canvas.height - this.axisHeight * r;\n        // console.debug(`WebLogo<${this.viewerId}>._calculate() maxHeight=${maxHeight}.`);\n        //#region Calculate screen\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const freq = this.positions[jPos].freq;\n            const rowCount = this.positions[jPos].rowCount;\n            const alphabetSize = this.getAlphabetSize();\n            if ((this.positionHeight == PositionHeight.Entropy) && (alphabetSize == null)) {\n                grok.shell.error('WebLogo: alphabet is undefined.');\n            }\n            const maxHeight = (this.positionHeight == PositionHeight.Entropy) ? (absoluteMaxHeight * (Math.log2(alphabetSize) - (this.positions[jPos].sumForHeightCalc)) / Math.log2(alphabetSize)) : absoluteMaxHeight;\n            let y = this.axisHeight * r + (absoluteMaxHeight - maxHeight - 1);\n            const entries = Object.entries(freq).sort((a, b) => {\n                if (a[0] !== '-' && b[0] !== '-')\n                    return b[1].count - a[1].count;\n                else if (a[0] === '-' && b[0] === '-')\n                    return 0;\n                else if (a[0] === '-')\n                    return -1;\n                else /* (b[0] === '-') */\n                    return +1;\n            });\n            for (const entry of entries) {\n                const pmInfo = entry[1];\n                // const m: string = entry[0];\n                const h = maxHeight * pmInfo.count / rowCount;\n                pmInfo.bounds = new DG.Rect(jPos * this.positionWidthWithMargin, y, this._positionWidth, h);\n                y += h;\n            }\n        }\n        //#endregion\n    }\n    // reflect changes made to filter/selection\n    render(recalc = true) {\n        var _a;\n        if (this.msgHost) {\n            if (this.seqCol && !this.cp) {\n                this.msgHost.innerText = `Unknown palette (column semType: '${this.seqCol.semType}').`;\n                this.msgHost.style.display = '';\n            }\n            else {\n                this.msgHost.style.display = 'none';\n            }\n        }\n        if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp)\n            return;\n        const g = this.canvas.getContext('2d');\n        if (!g)\n            return;\n        const r = window.devicePixelRatio;\n        if (recalc)\n            this._calculate(r);\n        g.resetTransform();\n        g.fillStyle = 'white';\n        g.fillRect(0, 0, this.canvas.width, this.canvas.height);\n        g.textBaseline = this.textBaseline;\n        // Prevents division by zero on Length = 0\n        if (this.startPosition === -1 || this.endPosition === -1)\n            return;\n        //#region Plot positionNames\n        const positionFontSize = 10 * r;\n        g.resetTransform();\n        g.fillStyle = 'black';\n        g.textAlign = 'center';\n        g.font = `${positionFontSize.toFixed(1)}px Roboto, Roboto Local, sans-serif`;\n        const posNameMaxWidth = Math.max(...this.positions.map((pos) => g.measureText(pos.name).width));\n        const hScale = posNameMaxWidth < (this._positionWidth - 2) ? 1 : (this._positionWidth - 2) / posNameMaxWidth;\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const pos = this.positions[jPos];\n            g.resetTransform();\n            g.setTransform(hScale, 0, 0, 1, jPos * this.positionWidthWithMargin + this._positionWidth / 2, 0);\n            g.fillText(pos.name, 0, 0);\n        }\n        //#endregion Plot positionNames\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {\n                if (monomer !== '-') {\n                    const monomerTxt = WebLogo.monomerToText(monomer);\n                    const b = pmInfo.bounds;\n                    const fontStyle = '16px Roboto, Roboto Local, sans-serif';\n                    // Hacks to scale uppercase characters to target rectangle\n                    const uppercaseLetterAscent = 0.25;\n                    const uppercaseLetterHeight = 12.2;\n                    g.resetTransform();\n                    g.strokeStyle = 'lightgray';\n                    g.lineWidth = 1;\n                    g.rect(b.left, b.top, b.width, b.height);\n                    g.fillStyle = (_a = this.cp.get(monomer)) !== null && _a !== void 0 ? _a : this.cp.get('other');\n                    g.textAlign = 'left';\n                    g.font = fontStyle;\n                    //g.fillRect(b.left, b.top, b.width, b.height);\n                    const mTm = g.measureText(monomerTxt);\n                    // if (mM.actualBoundingBoxAscent != 0)\n                    //   console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);\n                    g.setTransform(b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight, b.left, b.top);\n                    g.fillText(monomerTxt, 0, -uppercaseLetterAscent);\n                }\n            }\n        }\n    }\n    calcSize() {\n        if (!this.canvas || !this.host)\n            return;\n        const r = window.devicePixelRatio;\n        let width = this.Length * this.positionWidthWithMargin / r;\n        let height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));\n        if (this.fitArea) {\n            const xScale = this.root.clientHeight / height;\n            const yScale = this.root.clientWidth / width;\n            const scale = Math.max(1, Math.min(xScale, yScale));\n            width = width * scale;\n            height = height * scale;\n            this._positionWidth = this.positionWidth * scale;\n        }\n        this.canvas.width = width * r;\n        this.canvas.style.width = `${width}px`;\n        // const canvasHeight: number = width > this.root.clientWidth ? height - 8 : height;\n        this.host.style.setProperty('height', `${height}px`);\n        const canvasHeight = this.host.clientHeight;\n        this.canvas.height = canvasHeight * r;\n        this.canvas.style.setProperty('height', `${canvasHeight}px`);\n        // Adjust host and root width\n        if (this.fixWidth) {\n            // full width for canvas host and root\n            this.root.style.width = this.host.style.width = `${width}px`;\n            this.root.style.height = `${height}px`;\n            this.host.style.setProperty('overflow', 'hidden', 'important');\n        }\n        else {\n            // allow scroll canvas in root\n            this.root.style.width = this.host.style.width = '100%';\n            this.host.style.overflowX = 'auto!important';\n            this.host.style.setProperty('overflow', null);\n            this.host.style.setProperty('text-align', this.horizontalAlignment);\n            // vertical alignment\n            let hostTopMargin = 0;\n            switch (this.verticalAlignment) {\n                case 'top':\n                    hostTopMargin = 0;\n                    break;\n                case 'middle':\n                    hostTopMargin = Math.max(0, (this.root.clientHeight - height) / 2);\n                    break;\n                case 'bottom':\n                    hostTopMargin = Math.max(0, this.root.clientHeight - height);\n                    break;\n            }\n            this.host.style.setProperty('margin-top', `${hostTopMargin}px`, 'important');\n            if (this.root.clientHeight < height) {\n                this.host.style.setProperty('height', `${this.root.clientHeight}px`);\n                this.host.style.setProperty('overflow-y', null);\n            }\n            else {\n                this.host.style.setProperty('overflow-y', 'hidden', 'important');\n            }\n        }\n        // console.debug(\n        //   `this.root.style.height = ${this.root.style.height}\\n` +\n        //   `this.root.clientHeight = ${this.root.clientHeight}\\n` +\n        //   `this.host.style.height = ${this.host.style.height}\\n` +\n        //   `this.host.clientHeight = ${this.host.clientHeight}\\n` +\n        //   '\\n' +\n        //   `this.canvas.height       = ${this.canvas.height}\\n` +\n        //   `this.canvas.style.height = ${this.canvas.style.height}`);\n    }\n    /**\n     * @param {DG.Column} seqCol Column to look for a palette\n     * @param {number}  minLength minimum length of sequence to detect palette (empty strings are allowed)\n     * @return {SeqPalette} Palette corresponding to the alphabet of the sequences in the column\n     */\n    static pickUpPalette(seqCol, minLength = 5) {\n        let res = null;\n        switch (seqCol.semType) {\n            case Aminoacids.SemTypeMultipleAlignment:\n                res = AminoacidsPalettes.GrokGroups;\n                break;\n            case Nucleotides.SemTypeMultipleAlignment:\n                res = NucleotidesPalettes.Chromatogram;\n                break;\n        }\n        const stats = WebLogo.getStats(seqCol, minLength, WebLogo.splitterAsFasta);\n        const alphabetCandidates = [\n            [new Set(Object.keys(Nucleotides.Names)), NucleotidesPalettes.Chromatogram],\n            [new Set(Object.keys(Aminoacids.Names)), AminoacidsPalettes.GrokGroups],\n        ];\n        // Calculate likelihoods for alphabet_candidates\n        const alphabetCandidatesSim = alphabetCandidates\n            .map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[0]));\n        const maxCos = Math.max(...alphabetCandidatesSim);\n        if (maxCos > 0.55)\n            res = alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][1];\n        else\n            res = UnknownSeqPalettes.Color;\n        return res;\n    }\n    getAlphabetSize() {\n        var _a, _b;\n        return (_b = (_a = this.unitsHandler) === null || _a === void 0 ? void 0 : _a.getAlphabetSize()) !== null && _b !== void 0 ? _b : 0;\n    }\n    /** Stats of sequences with specified splitter func, returns { freq, sameLength }.\n     * @param {DG.Column} seqCol\n     * @param {number} minLength\n     * @param {SplitterFunc} splitter\n     * @return { SeqColStats }, sameLength: boolean } stats of column sequences\n     */\n    static getStats(seqCol, minLength, splitter) {\n        const freq = {};\n        let sameLength = true;\n        let firstLength = null;\n        for (const seq of seqCol.categories) {\n            const mSeq = splitter(seq);\n            if (firstLength == null)\n                firstLength = mSeq.length;\n            else if (mSeq.length !== firstLength)\n                sameLength = false;\n            if (mSeq.length > minLength) {\n                for (const m of mSeq) {\n                    if (!(m in freq))\n                        freq[m] = 0;\n                    freq[m] += 1;\n                }\n            }\n        }\n        return { freq: freq, sameLength: sameLength };\n    }\n    static getAlphabetSimilarity(freq, alphabet, gapSymbol = '-') {\n        const keys = new Set([...new Set(Object.keys(freq)), ...alphabet]);\n        keys.delete(gapSymbol);\n        const freqA = [];\n        const alphabetA = [];\n        for (const m of keys) {\n            freqA.push(m in freq ? freq[m] : 0);\n            alphabetA.push(alphabet.has(m) ? 1 : 0);\n        }\n        /* There were a few ideas: chi-squared, pearson correlation (variance?), scalar product */\n        const freqV = new Vector(freqA);\n        const alphabetV = new Vector(alphabetA);\n        return vectorDotProduct(freqV, alphabetV) / (vectorLength(freqV) * vectorLength(alphabetV));\n    }\n    // /** First try to find column with semType 'alignedSequence'.\n    //  * Next look for column with data alphabet corresponding to any of the known palettes.\n    //  * @param {DG.DataFrame} dataFrame\n    //  * @return {DG.Column} The column we were looking for or null\n    //  */\n    // private pickUpSeqCol(dataFrame: DG.DataFrame): DG.Column | null {\n    //   let res: DG.Column | null = dataFrame.columns.bySemType('alignedSequence');\n    //   if (res == null) {\n    //     for (const col of dataFrame.columns) {\n    //       const cp = WebLogo.pickUpPalette(col as DG.Column, 5);\n    //       if (cp !== null && !(cp instanceof UnknownSeqPalette)) {\n    //         res = col;\n    //         break;\n    //       }\n    //     }\n    //   }\n    //   return res;\n    // }\n    static pickUpSeqCol2(df) {\n        var _a;\n        const semTypeColList = df.columns.bySemTypeAll(DG.SEMTYPE.MACROMOLECULE);\n        let resCol = (_a = semTypeColList.find((col) => {\n            const units = col.getTag(DG.TAGS.UNITS);\n            return units ? units.indexOf('MSA') !== -1 : false;\n        })) !== null && _a !== void 0 ? _a : null;\n        if (!resCol && semTypeColList.length > 0)\n            resCol = semTypeColList[0];\n        return resCol;\n    }\n    /** Split sequence for single character monomers, square brackets multichar monomer names or gap symbol.\n     * @param {any} seq object with sequence\n     * @return {string[]} array of monomers\n     */\n    static splitterAsFasta(seq) {\n        const res = wu(seq.toString().matchAll(WebLogo.monomerRe))\n            .map((ma) => {\n            let mRes;\n            const m = ma[0];\n            if (m.length > 1) {\n                if (m in WebLogo.aaSynonyms) {\n                    mRes = WebLogo.aaSynonyms[m];\n                }\n                else {\n                    mRes = '';\n                    console.debug(`Long monomer '${m}' has not a short synonym.`);\n                }\n            }\n            else {\n                mRes = m;\n            }\n            return mRes;\n        }).toArray();\n        return res;\n    }\n    /** Splits Helm string to monomers, but does not replace monomer names to other notation (e.g. for RNA).\n     * Only for linear polymers, does not split RNA for ribose and phosphate monomers.\n     * @param {string} seq Source string of HELM notation\n     * @return {string[]}\n     */\n    static splitterAsHelm(seq) {\n        WebLogo.helmRe.lastIndex = 0;\n        const ea = WebLogo.helmRe.exec(seq.toString());\n        const inSeq = ea ? ea[2] : null;\n        const mmPostProcess = (mm) => {\n            WebLogo.helmPp1Re.lastIndex = 0;\n            const pp1M = WebLogo.helmPp1Re.exec(mm);\n            if (pp1M && pp1M.length >= 2)\n                return pp1M[1];\n            else\n                return mm;\n        };\n        const mmList = inSeq ? inSeq.split('.') : [];\n        const mmListRes = mmList.map(mmPostProcess);\n        return mmListRes;\n    }\n    /** Gets method to split sequence by separator\n     * @param {string} separator\n     * @param limit\n     * @return {SplitterFunc}\n     */\n    static getSplitterWithSeparator(separator, limit = undefined) {\n        return (seq) => {\n            return seq.split(separator, limit);\n        };\n    }\n    /** Get splitter method to split sequences to monomers\n     * @param {string} units\n     * @param {string} separator\n     * @param limit\n     * @return {SplitterFunc}\n     */\n    static getSplitter(units, separator, limit = undefined) {\n        if (units.toLowerCase().startsWith(\"fasta\" /* NOTATION.FASTA */))\n            return WebLogo.splitterAsFasta;\n        else if (units.toLowerCase().startsWith(\"separator\" /* NOTATION.SEPARATOR */))\n            return WebLogo.getSplitterWithSeparator(separator, limit);\n        else if (units.toLowerCase().startsWith(\"helm\" /* NOTATION.HELM */))\n            return WebLogo.splitterAsHelm;\n        else\n            throw new Error(`Unexpected units ${units} .`);\n        // TODO: Splitter for HELM\n    }\n    static getSplitterForColumn(col) {\n        if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n            throw new Error(`Get splitter for semType \"${DG.SEMTYPE.MACROMOLECULE}\" only.`);\n        const units = col.getTag(DG.TAGS.UNITS);\n        const separator = col.getTag(UnitsHandler.TAGS.separator);\n        return WebLogo.getSplitter(units, separator);\n    }\n    /* Shortens monomer representation text for monomers with long names.\n    **/\n    static monomerToText(src) {\n        const srcTxt = src.toString();\n        if (srcTxt.length <= 5) {\n            return srcTxt;\n        }\n        else {\n            const parts = wu(src.toString().matchAll(WebLogo.longMonomerPartRe))\n                .map((ma) => {\n                const mRes = ma[0];\n                return mRes;\n            }).toArray();\n            if (parts.length == 0) {\n                return ' ';\n            }\n            else {\n                const part0 = parts[0];\n                const resTxt = part0.length < 6 ? `${part0}…` : `${part0.substring(0, 5)}…`;\n                return resTxt;\n            }\n        }\n    }\n    static monomerToShort(amino, maxLengthOfMonomer) {\n        return amino.length <= maxLengthOfMonomer ? amino : amino.substring(0, maxLengthOfMonomer) + '…';\n    }\n}\nWebLogo.residuesSet = 'nucleotides';\nWebLogo.viewerCount = -1;\nWebLogo.monomerRe = /\\[(\\w+)\\]|(\\w)|(-)/g;\nWebLogo.helmRe = /(PEPTIDE1|DNA1|RNA1)\\{([^}]+)}/g;\nWebLogo.helmPp1Re = /\\[([^\\[\\]]+)]/g;\n/** Only some of the synonyms. These were obtained from the clustered oligopeptide dataset. */\nWebLogo.aaSynonyms = {\n    '[MeNle]': 'L',\n    '[MeA]': 'A',\n    '[MeG]': 'G',\n    '[MeF]': 'F',\n};\nWebLogo.longMonomerPartRe = /(\\w+)/g;\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"web-logo.js","sourceRoot":"","sources":["web-logo.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,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAE7B,OAAO,EAAC,MAAM,EAAC,MAAM,iDAAiD,CAAC;AACvE,OAAO,EAAC,YAAY,EAAE,gBAAgB,EAAC,MAAM,iDAAiD,CAAC;AAC/F,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAC,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAC,WAAW,EAAE,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAC,kBAAkB,EAAC,MAAM,YAAY,CAAC;AAG9C,OAAO,EAAW,YAAY,EAAC,MAAM,wBAAwB,CAAC;AAQ9D,IAAK,cAGJ;AAHD,WAAK,cAAc;IACjB,qCAAmB,CAAA;IACnB,+BAAa,CAAA;AACf,CAAC,EAHI,cAAc,KAAd,cAAc,QAGlB;AAYD;;;GAGG;AACH,iBAAiB,CAAC,SAAS,CAAC,iBAAiB,GAAG,UAAS,KAAiB,EAAE,CAAS;IACnF,MAAM,IAAI,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC1C,OAAO,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;AACvF,CAAC,CAAC;AAEF,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,GAAG,UAAS,CAAS,EAAE,CAAS;IACxD,OAAO,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC;AAChF,CAAC,CAAC;AAEF,MAAM,OAAO,mBAAmB;IAS9B,YAAY,QAAgB,CAAC,EAAE,SAAkB,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;CACF;AAED,MAAM,OAAO,YAAY;IAMvB;;;;;OAKG;IACH,YAAY,IAAY,EAAE,OAA6C,EAAE,EAAE,WAAmB,CAAC,EAAE,mBAA2B,CAAC;QAC3H,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,gBAAgB,GAAG,gBAAgB,CAAC;IAC3C,CAAC;CACF;AAED,MAAM,OAAO,OAAQ,SAAQ,EAAE,CAAC,QAAQ;IAyEtC;QACE,KAAK,EAAE,CAAC;QAtEF,aAAQ,GAAW,CAAC,CAAC,CAAC;QAEtB,gBAAW,GAAY,KAAK,CAAC;QAErC,4FAA4F;QAClF,OAAE,GAAsB,IAAI,CAAC;QAQ/B,eAAU,GAAW,EAAE,CAAC;QAExB,WAAM,GAA6B,IAAI,CAAC;QACxC,aAAQ,GAAwB,IAAI,CAAC;QAC7C,mCAAmC;QAC3B,cAAS,GAAmB,EAAE,CAAC;QAE/B,eAAU,GAAW,CAAC,CAAC;QACvB,aAAQ,GAAW,CAAC,CAAC;QAUtB,mBAAc,GAAW,CAAC,CAAC;QAW1B,kBAAa,GAAa,EAAE,CAAC;QAE7B,kBAAa,GAAW,CAAC,CAAC,CAAC;QAE3B,gBAAW,GAAW,CAAC,CAAC,CAAC;QAqBzB,aAAQ,GAAmB,EAAE,CAAC;QAKpC,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC;QACpC,OAAO,CAAC,WAAW,IAAI,CAAC,CAAC;QAEzB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QAEzB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,EAAE,CAAA;8DACnB,CAAC,CAAC;QACxD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,EAAE,CAAA;gEACa,CAAC,CAAC;QAC1D,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,GAAG,CAAA;gEACY,CAAC,CAAC;QAE1D,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,IAAI,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;QACvE,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAC;QAElE,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAChE,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,CAAC;QAE5D,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;QAE7C,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,QAAQ,EAChE,EAAC,OAAO,EAAE,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,EAAC,CAAC,CAAC;QAC1C,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,MAAM,CAAC,qBAAqB,EAAE,QAAQ,EACpE,EAAC,OAAO,EAAE,CAAC,MAAM,EAAE,QAAQ,EAAE,OAAO,CAAC,EAAC,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,CAAC;QAC1D,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,KAAK,CAAC,CAAC;QACjE,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,MAAM,CAAC,qBAAqB,EAAE,MAAM,EAClE,EAAC,OAAO,EAAE,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,EAAC,CAAC,CAAC;QACxC,IAAI,6BAA6B,GAAG,CAAC,CAAC;QACtC,IAAI,IAAI,CAAC,mBAAmB,KAAK,MAAM,EAAE;YACrC,6BAA6B,GAAG,CAAC,CAAC;SACrC;QACD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC,gBAAgB,EAAE,6BAA6B,EAAE,EAAC,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,EAAE,EAAC,CAAC,CAAC;QACnG,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,gBAAgB,EAAE,cAAc,CAAC,IAAI,EAAE,EAAC,OAAO,EAAE,CAAC,cAAc,CAAC,IAAI,EAAE,cAAc,CAAC,OAAO,CAAC,EAAC,CAAC,CAAC;IACrI,CAAC;IA5DD,0DAA0D;IAC1D,IAAY,MAAM;QAChB,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;SAC9B;QACD,OAAO,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAChG,CAAC;IAED,IAAY,uBAAuB;;QACjC,IAAI,CAAC,IAAI,CAAC,mBAAmB,KAAK,MAAM,CAAC,IAAI,CAAC,CAAA,MAAA,IAAI,CAAC,YAAY,0CAAE,sBAAsB,EAAE,MAAK,IAAI,CAAC,EAAE;YACnG,OAAO,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;SAClD;QACD,IAAI,IAAI,CAAC,mBAAmB,KAAK,QAAQ,EAAE;YACzC,OAAO,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;SAClD;QAED,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;IA6Ca,IAAI;;YAChB,IAAI,IAAI,CAAC,WAAW,EAAE;gBACpB,OAAO,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;gBAChD,OAAO;aACR;YAED,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;YACxB,IAAI,CAAC,OAAO,GAAG,qCAAqC,CAAC;YAErD,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YACpC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;YAEpC,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;YAC1B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YAEjC,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;YAEhD,6CAA6C;YAC7C,yCAAyC;YACzC,0CAA0C;YAE1C,MAAM,UAAU,GAAG,CAAC,CAAW,EAAuD,EAAE;gBACtF,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,CAAC;gBAC5D,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBAEtC,IAAI,QAAQ,KAAK,KAAK,CAAC;oBACrB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,MAAM,OAAO,GAAuB,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;qBAC3D,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAC3D,IAAI,OAAO,KAAK,SAAS;oBACvB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,OAAO,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YACjD,CAAC,CAAC;YAGF,qGAAqG;YACrG,OAAO;YAEP,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAa,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAa,EAAE,EAAE;gBAClG,IAAI,CAAC,IAAI,CAAC,MAAM;oBACd,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAE7B,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;gBAC1C,MAAM,OAAO,GAAa,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;gBACjE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;gBAE5C,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,IAAI,OAAO,EAAE;oBAC7D,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,CAAC,IAAY,EAAE,EAAE;wBAChF,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzE,OAAO,CAAC,CAAC,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,IAAI,KAAK,EAAE,IAAI,OAAO,KAAK,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;oBACrG,CAAC,CAAC,CAAC,MAAM,CAAS,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;oBACjD,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,OAAO,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,QAAQ,OAAO,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;iBACvG;qBAAM;oBACL,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;iBACnB;YACH,CAAC,CAAC,CAAC,CAAC;YAEJ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAa,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAa,EAAE,EAAE;gBAClG,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC;oBAC/B,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAC7B,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;gBAC1C,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;gBAE3E,uDAAuD;gBACvD,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,IAAI,OAAO,EAAE;oBAC7D,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE;wBACrC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzE,OAAO,CAAC,CAAC,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,IAAI,KAAK,EAAE,IAAI,OAAO,KAAK,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;oBACrG,CAAC,CAAC,CAAC;iBACJ;YACH,CAAC,CAAC,CAAC,CAAC;YAEJ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAE7F,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC5B,2CAA2C;YAE3C,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACpB,CAAC;KAAA;IAEO,iBAAiB,CAAC,IAAS;QACjC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAElB,iDAAiD;QACjD,qFAAqF;QACrF,iFAAiF;IACnF,CAAC;IAED;OACG;IACK,YAAY;QAClB,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YAC3F,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,EAAE;gBACvB,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACpD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;aACjE;YACD,IAAI,IAAI,CAAC,MAAM,EAAE;gBACf,MAAM,KAAK,GAAW,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACzD,MAAM,SAAS,GAAW,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBAC3E,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;gBACtD,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAElD,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,EAAE,GAAG,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;aAC9C;iBAAM;gBACL,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;gBACrB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;gBACxB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;gBACxB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC;gBACtB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;aAChB;SACF;QACD,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAEO,eAAe;QACrB,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,OAAO;QAET,IAAI,UAA8B,CAAC;QACnC,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,MAAM,OAAO,GAAe,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACvE,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAC7B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SAC/F;aAAM;YACL,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC;SACrC;QACD,MAAM,SAAS,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,GAAG,CAClE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzD,0DAA0D;QAC1D,MAAM,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAC7D,kDAAkD;QAClD,IAAI,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAAC,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACzF,CAAC,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,EAAE,CAAC,CAAC;QAE5D,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,aAAa;YAChE,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;YACtD,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACzD,IAAI,CAAC,WAAW,GAAG,CAAC,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,aAAa;YAC5D,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;YACpD,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;IACvE,CAAC;IAEe,iBAAiB,CAAC,QAAqB;QACrD,wEAAwE;QACxE,KAAK,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;QAElC,QAAQ,QAAQ,CAAC,IAAI,EAAE;YACvB,KAAK,uBAAuB;gBAC1B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,oBAAoB;gBACvB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,mBAAmB;gBACtB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,iBAAiB;gBACpB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,eAAe;gBAClB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC;gBACzC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,UAAU;gBACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,mBAAmB;gBACtB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,qBAAqB;gBACxB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,SAAS;gBACZ,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,iBAAiB;gBACpB,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,oBAAoB;gBACvB,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,gBAAgB;gBACnB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,qBAAqB;gBACxB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,gBAAgB;gBACnB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;SACP;IACH,CAAC;IAEqB,eAAe;;;;;YACnC,MAAM,YAAY,GAAW,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;YAC9D,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,kCAAkC,YAAY,UAAU,CAAC,CAAC;YACrG,OAAM,eAAe,YAAG;YAExB,IAAI,CAAC,YAAY,EAAE,CAAC;YAEpB,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,CAAC,EAAE;gBAC7B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBACnF,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;oBAC7D,IAAI,CAAC,eAAe,EAAE,CAAC;oBACvB,IAAI,CAAC,MAAM,EAAE,CAAC;gBAChB,CAAC,CAAC,CAAC,CAAC;aACL;YAED,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,yBAAyB,CAAC,CAAC;QACxE,CAAC;KAAA;IAEqB,MAAM;;;;;YAC1B,MAAM,YAAY,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;YAC3D,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,kCAAkC,YAAY,UAAU,CAAC,CAAC;YACrG,OAAM,MAAM,YAAG;YAEf,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;YAClD,IAAI,CAAC,IAAK,CAAC,MAAM,EAAE,CAAC;YACpB,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;YACxB,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;YACzB,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC;YAEtB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACzB,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,yBAAyB,CAAC,CAAC;QACxE,CAAC;KAAA;IAES,aAAa,CAAC,aAAa,GAAG,GAAG;QACzC,IAAI,IAAI,CAAC,qBAAqB;YAC5B,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAE7D,OAAO,EAAE,CAAC;IACZ,CAAC;IAES,WAAW,CAAC,KAAiB,EAAE,SAA8B;QACrE,IAAI,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC1B,IAAI,cAAc,GAAG,CAAC,CAAC;QACvB,KAAK,IAAI,cAAc,GAAG,CAAC,EAAE,cAAc,GAAG,MAAM,EAAE,cAAc,EAAE,EAAE;YACtE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC,EAAE;gBACrC,KAAK,CAAC,cAAc,CAAC,GAAG,KAAK,CAAC,cAAc,CAAC,CAAC;gBAC9C,cAAc,EAAE,CAAC;aAClB;SACF;QACD,KAAK,CAAC,MAAM,GAAG,cAAc,CAAC;QAC9B,OAAO,KAAK,CAAC;IACf,CAAC;IAGS,qBAAqB;QAC7B,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,EAAE,WAAC,OAAA,CAAA,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,IAAI,CAAC,GAAG,CAAC,0CAAE,KAAK,MAAK,IAAI,CAAC,QAAQ,CAAA,EAAA,CAAC,CAAC;SACpF;IACH,CAAC;IAES,UAAU,CAAC,CAAS;QAC5B,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS;YAC/D,OAAO;QACT,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAGlD,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEhB,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,CAAC,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnH,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,OAAO,GAAW,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC;YACtE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,YAAY,CAAC,OAAO,CAAC,CAAC;SAClD;QAED,iFAAiF;QACjF,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;QAC3D,2GAA2G;QAC3G,gDAAgD;QAEhD,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;QAElB,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE;YACvB,IAAI,CAAC,GAAmB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YAE7C,IAAI,CAAC,CAAC,EAAE;gBACN,CAAC,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;gBACzB,EAAE,IAAI,CAAC,QAAQ,CAAC;aACjB;YAED,MAAM,IAAI,GAAa,IAAI,CAAC,QAAS,CAAC,CAAC,CAAC,CAAC;YACzC,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;gBAC7C,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;gBACzC,MAAM,CAAC,GAAW,IAAI,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,IAAI,GAAG,CAAC;gBACzD,IAAI,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC;oBAChB,MAAM,CAAC,CAAC,CAAC,GAAG,IAAI,mBAAmB,EAAE,CAAC;gBACxC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC;aACnB;SACF;QAED,4BAA4B;QAC5B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,yCAAyC;YAEzC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC;YAClC,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI;gBACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YACtE,IAAI,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,EAAE;gBACjD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,GAAG,CAAC,CAAC;gBAC1C,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE;oBACzC,MAAM,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC;oBAC9E,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;iBAC9D;aACF;SACF;QACD,YAAY;QACZ,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAE7B,MAAM,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;QACnE,mFAAmF;QAEnF,0BAA0B;QAC1B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,IAAI,GAAyC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC;YAC/C,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;YAC5C,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,EAAE;gBAC7E,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;aACrD;YAED,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;YAE5M,IAAI,CAAC,GAAW,IAAI,CAAC,UAAU,GAAG,CAAC,GAAG,CAAC,iBAAiB,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC;YAE1E,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBACjD,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBAC9B,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;qBAC5B,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnC,OAAO,CAAC,CAAC;qBACN,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnB,OAAO,CAAC,CAAC,CAAC;qBACP,oBAAoB;oBACvB,OAAO,CAAC,CAAC,CAAC;YACd,CAAC,CAAC,CAAC;YACH,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE;gBAC3B,MAAM,MAAM,GAAwB,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC7C,8BAA8B;gBAC9B,MAAM,CAAC,GAAW,SAAS,GAAG,MAAM,CAAC,KAAK,GAAG,QAAQ,CAAC;gBAEtD,MAAM,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC,EAAE,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;gBAC5F,CAAC,IAAI,CAAC,CAAC;aACR;SACF;QACD,YAAY;IACd,CAAC;IAED,2CAA2C;IAC3C,MAAM,CAAC,MAAM,GAAG,IAAI;;QAClB,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;gBAC3B,IAAI,CAAC,OAAQ,CAAC,SAAS,GAAG,qCAAqC,IAAI,CAAC,MAAM,CAAC,OAAO,KAAK,CAAC;gBACxF,IAAI,CAAC,OAAQ,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC;aAClC;iBAAM;gBACL,IAAI,CAAC,OAAQ,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;aACtC;SACF;QAED,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,EAAE;YAC7D,OAAO;QAET,MAAM,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC,CAAC;YAAE,OAAO;QAEf,MAAM,CAAC,GAAG,MAAM,CAAC,gBAAgB,CAAC;QAElC,IAAI,MAAM;YACR,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAErB,CAAC,CAAC,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC;QACtB,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACxD,CAAC,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QAEnC,0CAA0C;QAC1C,IAAI,IAAI,CAAC,aAAa,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,KAAK,CAAC,CAAC;YACtD,OAAO;QAET,4BAA4B;QAC5B,MAAM,gBAAgB,GAAG,EAAE,GAAG,CAAC,CAAC;QAChC,CAAC,CAAC,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC;QACtB,CAAC,CAAC,SAAS,GAAG,QAAQ,CAAC;QACvB,CAAC,CAAC,IAAI,GAAG,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC,CAAC,qCAAqC,CAAC;QAC7E,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QAChG,MAAM,MAAM,GAAG,eAAe,GAAG,CAAC,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC,GAAG,eAAe,CAAC;QAE7G,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,GAAG,GAAiB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC/C,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,CAAC,CAAC,YAAY,CACZ,MAAM,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EACf,IAAI,GAAG,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,cAAc,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YACpE,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;SAC5B;QACD,+BAA+B;QAE/B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,KAAK,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE;gBACzE,IAAI,OAAO,KAAK,GAAG,EAAE;oBACnB,MAAM,UAAU,GAAG,OAAO,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;oBAClD,MAAM,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;oBAExB,MAAM,SAAS,GAAG,uCAAuC,CAAC;oBAC1D,0DAA0D;oBAC1D,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBACnC,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBAEnC,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,WAAW,GAAG,WAAW,CAAC;oBAC5B,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;oBAChB,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC;oBACzC,CAAC,CAAC,SAAS,GAAG,MAAA,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,mCAAI,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;oBAC3D,CAAC,CAAC,SAAS,GAAG,MAAM,CAAC;oBACrB,CAAC,CAAC,IAAI,GAAG,SAAS,CAAC;oBACnB,+CAA+C;oBAC/C,MAAM,GAAG,GAAgB,CAAC,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;oBAEnD,uCAAuC;oBACvC,yFAAyF;oBAEzF,CAAC,CAAC,YAAY,CACZ,CAAC,CAAC,KAAK,GAAG,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,MAAM,GAAG,qBAAqB,EAC3D,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC;oBACjB,CAAC,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC,EAAE,CAAC,qBAAqB,CAAC,CAAC;iBACnD;aACF;SACF;IACH,CAAC;IAEO,QAAQ;QACd,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI;YAC5B,OAAO;QAET,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;QAE1C,IAAI,KAAK,GAAW,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,uBAAuB,GAAG,CAAC,CAAC;QACnE,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;QAExF,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,MAAM,MAAM,GAAW,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC;YACvD,MAAM,MAAM,GAAW,IAAI,CAAC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACrD,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC;YACpD,KAAK,GAAG,KAAK,GAAG,KAAK,CAAC;YACtB,MAAM,GAAG,MAAM,GAAG,KAAK,CAAC;YACxB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;SAClD;QAED,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,GAAG,CAAC,CAAC;QAC9B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAC;QAEvC,oFAAoF;QACpF,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,MAAM,IAAI,CAAC,CAAC;QACrD,MAAM,YAAY,GAAW,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC;QACpD,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,YAAY,GAAG,CAAC,CAAC;QACtC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,YAAY,IAAI,CAAC,CAAC;QAE7D,6BAA6B;QAC7B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,sCAAsC;YACtC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAC;YAC7D,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC;YACvC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,UAAU,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;SAChE;aAAM;YACL,8BAA8B;YAC9B,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YACvD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,gBAAgB,CAAC;YAC7C,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC;YAE9C,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,IAAI,CAAC,mBAAmB,CAAC,CAAC;YAEpE,qBAAqB;YACrB,IAAI,aAAa,GAAG,CAAC,CAAC;YACtB,QAAQ,IAAI,CAAC,iBAAiB,EAAE;gBAChC,KAAK,KAAK;oBACR,aAAa,GAAG,CAAC,CAAC;oBAClB,MAAM;gBACR,KAAK,QAAQ;oBACX,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;oBACnE,MAAM;gBACR,KAAK,QAAQ;oBACX,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,CAAC;oBAC7D,MAAM;aACP;YACD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,GAAG,aAAa,IAAI,EAAE,WAAW,CAAC,CAAC;YAE7E,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,EAAE;gBACnC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,CAAC;gBACrE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;aACjD;iBAAM;gBACL,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;aAClE;SACF;QAED,iBAAiB;QACjB,6DAA6D;QAC7D,6DAA6D;QAC7D,6DAA6D;QAC7D,6DAA6D;QAC7D,WAAW;QACX,2DAA2D;QAC3D,+DAA+D;IACjE,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,aAAa,CAAC,MAAiB,EAAE,YAAoB,CAAC;QAClE,IAAI,GAAG,GAAsB,IAAI,CAAC;QAClC,QAAQ,MAAM,CAAC,OAAO,EAAE;YACxB,KAAK,UAAU,CAAC,wBAAwB;gBACtC,GAAG,GAAG,kBAAkB,CAAC,UAAU,CAAC;gBACpC,MAAM;YACR,KAAK,WAAW,CAAC,wBAAwB;gBACvC,GAAG,GAAG,mBAAmB,CAAC,YAAY,CAAC;gBACvC,MAAM;SACP;QACD,MAAM,KAAK,GAAgB,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC;QAExF,MAAM,kBAAkB,GAAgC;YACtD,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,mBAAmB,CAAC,YAAY,CAAC;YAC3E,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,kBAAkB,CAAC,UAAU,CAAC;SACxE,CAAC;QACF,gDAAgD;QAChD,MAAM,qBAAqB,GAAa,kBAAkB;aACvD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,qBAAqB,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,qBAAqB,CAAC,CAAC;QAClD,IAAI,MAAM,GAAG,IAAI;YACf,GAAG,GAAG,kBAAkB,CAAC,qBAAqB,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;;YAEnE,GAAG,GAAG,kBAAkB,CAAC,KAAK,CAAC;QAEjC,OAAO,GAAG,CAAC;IACb,CAAC;IAEM,eAAe;;QACpB,OAAO,MAAA,MAAA,IAAI,CAAC,YAAY,0CAAE,eAAe,EAAE,mCAAI,CAAC,CAAC;IACnD,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,QAAQ,CAAC,MAAiB,EAAE,SAAiB,EAAE,QAAsB;QAC1E,MAAM,IAAI,GAA4B,EAAE,CAAC;QACzC,IAAI,UAAU,GAAG,IAAI,CAAC;QACtB,IAAI,WAAW,GAAG,IAAI,CAAC;QAEvB,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,UAAU,EAAE;YACnC,MAAM,IAAI,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;YAE3B,IAAI,WAAW,IAAI,IAAI;gBACrB,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC;iBACvB,IAAI,IAAI,CAAC,MAAM,KAAK,WAAW;gBAClC,UAAU,GAAG,KAAK,CAAC;YAErB,IAAI,IAAI,CAAC,MAAM,GAAG,SAAS,EAAE;gBAC3B,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE;oBACpB,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;wBACd,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;oBACd,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;iBACd;aACF;SACF;QACD,OAAO,EAAC,IAAI,EAAE,IAAI,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC;IAC9C,CAAC;IAEM,MAAM,CAAC,qBAAqB,CAAC,IAAkB,EAAE,QAAqB,EAAE,YAAoB,GAAG;QACpG,MAAM,IAAI,GAAG,IAAI,GAAG,CAAS,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;QAC3E,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAEvB,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,MAAM,SAAS,GAAa,EAAE,CAAC;QAC/B,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE;YACpB,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACpC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SACzC;QACD,0FAA0F;QAC1F,MAAM,KAAK,GAAW,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;QACxC,MAAM,SAAS,GAAW,IAAI,MAAM,CAAC,SAAS,CAAC,CAAC;QAChD,OAAO,gBAAgB,CAAC,KAAK,EAAE,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC;IAC9F,CAAC;IAED,+DAA+D;IAC/D,yFAAyF;IACzF,qCAAqC;IACrC,gEAAgE;IAChE,MAAM;IACN,oEAAoE;IACpE,gFAAgF;IAChF,uBAAuB;IACvB,6CAA6C;IAC7C,+DAA+D;IAC/D,iEAAiE;IACjE,qBAAqB;IACrB,iBAAiB;IACjB,UAAU;IACV,QAAQ;IACR,MAAM;IACN,gBAAgB;IAChB,IAAI;IAEG,MAAM,CAAC,aAAa,CAAC,EAAgB;;QAC1C,MAAM,cAAc,GAAG,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QACzE,IAAI,MAAM,GAAqB,MAAA,cAAc,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;YACzD,MAAM,KAAK,GAAG,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACxC,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QACrD,CAAC,CAAC,mCAAI,IAAI,CAAC;QACX,IAAI,CAAC,MAAM,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC;YACtC,MAAM,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;QAC7B,OAAO,MAAM,CAAC;IAChB,CAAC;IAID;;;OAGG;IACI,MAAM,CAAC,eAAe,CAAC,GAAQ;QACpC,MAAM,GAAG,GAAa,EAAE,CAAmB,GAAG,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;aACnF,GAAG,CAAC,CAAC,EAAoB,EAAE,EAAE;YAC5B,IAAI,IAAY,CAAC;YACjB,MAAM,CAAC,GAAW,EAAE,CAAC,CAAC,CAAC,CAAC;YACxB,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;gBAChB,IAAI,CAAC,IAAI,OAAO,CAAC,UAAU,EAAE;oBAC3B,IAAI,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;iBAC9B;qBAAM;oBACL,IAAI,GAAG,EAAE,CAAC;oBACV,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,4BAA4B,CAAC,CAAC;iBAC/D;aACF;iBAAM;gBACL,IAAI,GAAG,CAAC,CAAC;aACV;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;QAEf,OAAO,GAAG,CAAC;IACb,CAAC;IAKD;;;;OAIG;IACI,MAAM,CAAC,cAAc,CAAC,GAAQ;QACnC,OAAO,CAAC,MAAM,CAAC,SAAS,GAAG,CAAC,CAAC;QAC7B,MAAM,EAAE,GAA2B,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;QACvE,MAAM,KAAK,GAAkB,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAE/C,MAAM,aAAa,GAAG,CAAC,EAAU,EAAU,EAAE;YAC3C,OAAO,CAAC,SAAS,CAAC,SAAS,GAAG,CAAC,CAAC;YAChC,MAAM,IAAI,GAAG,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACxC,IAAI,IAAI,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC;gBAC1B,OAAO,IAAI,CAAC,CAAC,CAAC,CAAC;;gBAEf,OAAO,EAAE,CAAC;QACd,CAAC,CAAC;QAEF,MAAM,MAAM,GAAa,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACvD,MAAM,SAAS,GAAa,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;QACtD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,wBAAwB,CAAC,SAAiB,EAAE,QAA4B,SAAS;QAC7F,OAAO,CAAC,GAAW,EAAE,EAAE;YACrB,OAAO,GAAG,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;QACrC,CAAC,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACI,MAAM,CAAC,WAAW,CAAC,KAAa,EAAE,SAAiB,EAAE,QAA4B,SAAS;QAC/F,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,8BAAgB;YAChD,OAAO,OAAO,CAAC,eAAe,CAAC;aAC5B,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,sCAAoB;YACzD,OAAO,OAAO,CAAC,wBAAwB,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;aACvD,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,4BAAe;YACpD,OAAO,OAAO,CAAC,cAAc,CAAC;;YAE9B,MAAM,IAAI,KAAK,CAAC,oBAAoB,KAAK,IAAI,CAAC,CAAC;QAEjD,0BAA0B;IAC5B,CAAC;IAEM,MAAM,CAAC,oBAAoB,CAAC,GAAc;QAC/C,IAAI,GAAG,CAAC,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC,aAAa;YAC1C,MAAM,IAAI,KAAK,CAAC,6BAA6B,EAAE,CAAC,OAAO,CAAC,aAAa,SAAS,CAAC,CAAC;QAElF,MAAM,KAAK,GAAG,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxC,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1D,OAAO,OAAO,CAAC,WAAW,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IAC/C,CAAC;IAYD;OACG;IACI,MAAM,CAAC,aAAa,CAAC,GAAQ;QAClC,MAAM,MAAM,GAAW,GAAG,CAAC,QAAQ,EAAE,CAAC;QACtC,IAAI,MAAM,CAAC,MAAM,IAAI,CAAC,EAAE;YACtB,OAAO,MAAM,CAAC;SACf;aAAM;YACL,MAAM,KAAK,GAAa,EAAE,CAAmB,GAAG,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;iBAC7F,GAAG,CAAC,CAAC,EAAoB,EAAE,EAAE;gBAC5B,MAAM,IAAI,GAAW,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC3B,OAAO,IAAI,CAAC;YACd,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;YAEf,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,EAAE;gBACrB,OAAO,GAAG,CAAC;aACZ;iBAAM;gBACL,MAAM,KAAK,GAAW,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC/B,MAAM,MAAM,GAAW,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC;gBACpF,OAAO,MAAM,CAAC;aACf;SACF;IACH,CAAC;IAEM,MAAM,CAAC,cAAc,CAAC,KAAa,EAAE,kBAA0B;QACpE,OAAO,KAAK,CAAC,MAAM,IAAI,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,kBAAkB,CAAC,GAAG,GAAG,CAAC;IACnG,CAAC;;AAh3Ba,mBAAW,GAAG,aAAa,CAAC;AAC3B,mBAAW,GAAW,CAAC,CAAC,CAAC;AA+uBzB,iBAAS,GAAG,qBAAqB,CAAC;AA2BlC,cAAM,GAAG,iCAAiC,CAAC;AAC3C,iBAAS,GAAG,gBAAgB,CAAC;AAiE5C,8FAA8F;AAC/E,kBAAU,GAA+B;IACtD,SAAS,EAAE,GAAG;IACd,OAAO,EAAE,GAAG;IACZ,OAAO,EAAE,GAAG;IACZ,OAAO,EAAE,GAAG;CACb,CAAC;AAEa,yBAAiB,GAAG,QAAQ,CAAC","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport wu from 'wu';\nimport * as rxjs from 'rxjs';\n\nimport {Vector} from '@datagrok-libraries/utils/src/type-declarations';\nimport {vectorLength, vectorDotProduct} from '@datagrok-libraries/utils/src/vector-operations';\nimport {Aminoacids, AminoacidsPalettes} from '../aminoacids';\nimport {Nucleotides, NucleotidesPalettes} from '../nucleotides';\nimport {UnknownSeqPalettes} from '../unknown';\nimport {SeqPalette} from '../seq-palettes';\nimport {Subscription} from 'rxjs';\nimport {NOTATION, UnitsHandler} from '../utils/units-handler';\n\ndeclare module 'datagrok-api/src/grid' {\n  interface Rect {\n    contains(x: number, y: number): boolean;\n  }\n}\n\nenum PositionHeight {\n  Entropy = 'Entropy',\n  full = '100%',\n}\n\ndeclare global {\n  interface HTMLCanvasElement {\n    getCursorPosition(event: MouseEvent, r: number): DG.Point;\n  }\n}\n\nexport type MonomerFreqs = { [m: string]: number };\nexport type SeqColStats = { freq: MonomerFreqs, sameLength: boolean }\nexport type SplitterFunc = (seq: string) => string[];\n\n/**@param {MouseEvent} event\n * @param {number} r devicePixelRation\n * @return {DG.Point} canvas related cursor position\n */\nHTMLCanvasElement.prototype.getCursorPosition = function(event: MouseEvent, r: number): DG.Point {\n  const rect = this.getBoundingClientRect();\n  return new DG.Point((event.clientX - rect.left) * r, (event.clientY - rect.top) * r);\n};\n\nDG.Rect.prototype.contains = function(x: number, y: number): boolean {\n  return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;\n};\n\nexport class PositionMonomerInfo {\n  /** Sequences count with monomer in position\n   */\n  count: number;\n\n  /** Remember screen coords rect\n   */\n  bounds: DG.Rect;\n\n  constructor(count: number = 0, bounds: DG.Rect = new DG.Rect(0, 0, 0, 0)) {\n    this.count = count;\n    this.bounds = bounds;\n  }\n}\n\nexport class PositionInfo {\n  public readonly name: string;\n  freq: { [m: string]: PositionMonomerInfo };\n  rowCount: number;\n  sumForHeightCalc: number;\n\n  /** freq = {}, rowCount = 0\n   * @param {string} name Name of position ('111A', '111.1', etc)\n   * @param {number} sumForHeightCalc Sum of all monomer counts for height calculation\n   * @param {number} rowCount Count of elements in column\n   * @param {string[]} freq frequency of monomers in position\n   */\n  constructor(name: string, freq: { [m: string]: PositionMonomerInfo } = {}, rowCount: number = 0, sumForHeightCalc: number = 0) {\n    this.name = name;\n    this.freq = freq;\n    this.rowCount = rowCount;\n    this.sumForHeightCalc = sumForHeightCalc;\n  }\n}\n\nexport class WebLogo extends DG.JsViewer {\n  public static residuesSet = 'nucleotides';\n  private static viewerCount: number = -1;\n\n  private viewerId: number = -1;\n  private unitsHandler: UnitsHandler | null;\n  private initialized: boolean = false;\n\n  // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];\n  protected cp: SeqPalette | null = null;\n\n  private host?: HTMLDivElement;\n  private msgHost?: HTMLElement;\n  private canvas?: HTMLCanvasElement;\n  private slider?: DG.RangeSlider;\n  private textBaseline: CanvasTextBaseline;\n\n  private axisHeight: number = 12;\n\n  private seqCol: DG.Column<string> | null = null;\n  private splitter: SplitterFunc | null = null;\n  // private maxLength: number = 100;\n  private positions: PositionInfo[] = [];\n\n  private rowsMasked: number = 0;\n  private rowsNull: number = 0;\n\n  // Viewer's properties (likely they should be public so that they can be set outside)\n  private _positionWidth: number;\n  public positionWidth: number;\n  public minHeight: number;\n  public maxHeight: number;\n  public considerNullSequences: boolean;\n  public sequenceColumnName: string | null;\n  public positionMarginState: string;\n  public positionMargin: number = 0;\n  public startPositionName: string | null;\n  public endPositionName: string | null;\n  public fixWidth: boolean;\n  public verticalAlignment: string | null;\n  public horizontalAlignment: string | null;\n  public fitArea: boolean;\n  public shrinkEmptyTail: boolean;\n  public skipEmptyPositions: boolean;\n  public positionHeight: string;\n\n  private positionNames: string[] = [];\n\n  private startPosition: number = -1;\n\n  private endPosition: number = -1;\n\n  /** For startPosition equals to endPosition Length is 1 */\n  private get Length(): number {\n    if (this.skipEmptyPositions) {\n      return this.positions.length;\n    }\n    return this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0;\n  }\n\n  private get positionWidthWithMargin() {\n    if ((this.positionMarginState === 'auto') && (this.unitsHandler?.getAlphabetIsMultichar() === true)) {\n      return this._positionWidth + this.positionMargin;\n    }\n    if (this.positionMarginState === 'enable') {\n      return this._positionWidth + this.positionMargin;\n    }\n\n    return this._positionWidth;\n  }\n\n  private viewSubs: Subscription[] = [];\n\n  constructor() {\n    super();\n\n    this.viewerId = WebLogo.viewerCount;\n    WebLogo.viewerCount += 1;\n\n    this.textBaseline = 'top';\n    this.unitsHandler = null;\n\n    this._positionWidth = this.positionWidth = this.float('positionWidth', 16/*,\n      {editor: 'slider', min: 4, max: 64, postfix: 'px'}*/);\n    this.minHeight = this.float('minHeight', 50/*,\n      {editor: 'slider', min: 25, max: 250, postfix: 'px'}*/);\n    this.maxHeight = this.float('maxHeight', 100/*,\n      {editor: 'slider', min: 25, max: 500, postfix: 'px'}*/);\n\n    this.considerNullSequences = this.bool('considerNullSequences', false);\n    this.sequenceColumnName = this.string('sequenceColumnName', null);\n\n    this.startPositionName = this.string('startPositionName', null);\n    this.endPositionName = this.string('endPositionName', null);\n\n    this.fixWidth = this.bool('fixWidth', false);\n\n    this.verticalAlignment = this.string('verticalAlignment', 'middle',\n      {choices: ['top', 'middle', 'bottom']});\n    this.horizontalAlignment = this.string('horizontalAlignment', 'center',\n      {choices: ['left', 'center', 'right']});\n    this.fitArea = this.bool('fitArea', true);\n    this.shrinkEmptyTail = this.bool('shrinkEmptyTail', true);\n    this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n    this.positionMarginState = this.string('positionMarginState', 'auto',\n      {choices: ['auto', 'enable', 'off']});\n    let defaultValueForPositionMargin = 0;\n    if (this.positionMarginState === 'auto') {\n        defaultValueForPositionMargin = 4;\n    }\n    this.positionMargin = this.int('positionMargin', defaultValueForPositionMargin, {min: 0, max: 16});\n    this.positionHeight = this.string('positionHeight', PositionHeight.full, {choices: [PositionHeight.full, PositionHeight.Entropy]});\n  }\n\n  private async init(): Promise<void> {\n    if (this.initialized) {\n      console.error('WebLogo second initialization!');\n      return;\n    }\n\n    this.initialized = true;\n    this.helpUrl = '/help/visualize/viewers/web-logo.md';\n\n    this.msgHost = ui.div('No message');\n    this.msgHost.style.display = 'none';\n\n    this.canvas = ui.canvas();\n    this.canvas.style.width = '100%';\n\n    this.host = ui.div([this.msgHost, this.canvas]);\n\n    // this.slider = ui.rangeSlider(0, 20, 2, 5);\n    // this.slider.root.style.width = '100%';\n    // this.slider.root.style.height = '12px';\n\n    const getMonomer = (p: DG.Point): [number, string | null, PositionMonomerInfo | null] => {\n      const jPos = Math.floor(p.x / this.positionWidthWithMargin);\n      const position = this.positions[jPos];\n\n      if (position === void 0)\n        return [jPos, null, null];\n\n      const monomer: string | undefined = Object.keys(position.freq)\n        .find((m) => position.freq[m].bounds.contains(p.x, p.y));\n      if (monomer === undefined)\n        return [jPos, null, null];\n\n      return [jPos, monomer, position.freq[monomer]];\n    };\n\n\n    // this.subs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mouseover').subscribe((e: MouseEvent) => {\n    // }));\n\n    this.viewSubs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mousemove').subscribe((e: MouseEvent) => {\n      if (!this.canvas)\n        return;\n\n      const args = e as MouseEvent;\n\n      const r: number = window.devicePixelRatio;\n      const cursorP: DG.Point = this.canvas.getCursorPosition(args, r);\n      const [jPos, monomer] = getMonomer(cursorP);\n\n      if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n        const rowCount = wu.count().take(this.dataFrame.rowCount).filter((iRow: number) => {\n          const seq = this.seqCol!.get(iRow);\n          const seqM = seq ? this.splitter!(seq)[this.startPosition + jPos] : null;\n          return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n        }).reduce<number>((count, iRow) => count + 1, 0);\n        ui.tooltip.show(ui.div([ui.div(`${monomer}`), ui.div(`${rowCount} rows`)]), args.x + 16, args.y + 16);\n      } else {\n        ui.tooltip.hide();\n      }\n    }));\n\n    this.viewSubs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mousedown').subscribe((e: MouseEvent) => {\n      if (!this.canvas || e.button != 0)\n        return;\n\n      const args = e as MouseEvent;\n      const r: number = window.devicePixelRatio;\n      const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args, r));\n\n      // prevents deselect all rows if we miss monomer bounds\n      if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n        this.dataFrame.selection.init((iRow) => {\n          const seq = this.seqCol!.get(iRow);\n          const seqM = seq ? this.splitter!(seq)[this.startPosition + jPos] : null;\n          return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n        });\n      }\n    }));\n\n    this.viewSubs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n\n    this.root.append(this.host);\n    // this.root.appendChild(this.slider.root);\n\n    this.render(true);\n  }\n\n  private rootOnSizeChanged(args: any) {\n    this.render(true);\n\n    // console.debug(`WebLogo.onRootSizeChanged() ` +\n    //   `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +\n    //   `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);\n  }\n\n  /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().\n   */\n  private updateSeqCol(): void {\n    if (this.dataFrame) {\n      this.seqCol = this.sequenceColumnName ? this.dataFrame.col(this.sequenceColumnName) : null;\n      if (this.seqCol == null) {\n        this.seqCol = WebLogo.pickUpSeqCol2(this.dataFrame);\n        this.sequenceColumnName = this.seqCol ? this.seqCol.name : null;\n      }\n      if (this.seqCol) {\n        const units: string = this.seqCol!.getTag(DG.TAGS.UNITS);\n        const separator: string = this.seqCol!.getTag(UnitsHandler.TAGS.separator);\n        this.splitter = WebLogo.getSplitter(units, separator);\n        this.unitsHandler = new UnitsHandler(this.seqCol);\n\n        this.updatePositions();\n        this.cp = WebLogo.pickUpPalette(this.seqCol);\n      } else {\n        this.splitter = null;\n        this.positionNames = [];\n        this.startPosition = -1;\n        this.endPosition = -1;\n        this.cp = null;\n      }\n    }\n    this.render();\n  }\n\n  private updatePositions(): void {\n    if (!this.seqCol)\n      return;\n\n    let categories: (string | null) [];\n    if (this.shrinkEmptyTail) {\n      const indices: Int32Array = this.dataFrame.filter.getSelectedIndexes();\n      categories = Array.from(new Set(\n        Array.from(Array(indices.length).keys()).map((i: number) => this.seqCol!.get(indices[i]))));\n    } else {\n      categories = this.seqCol.categories;\n    }\n    const maxLength = categories.length > 0 ? Math.max(...categories.map(\n      (s) => s !== null ? this.splitter!(s).length : 0)) : 0;\n\n    // Get position names from data column tag 'positionNames'\n    const positionNamesTxt = this.seqCol.getTag('positionNames');\n    // Fallback if 'positionNames' tag is not provided\n    this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :\n      [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);\n\n    this.startPosition = (this.startPositionName && this.positionNames &&\n      this.positionNames.includes(this.startPositionName)) ?\n      this.positionNames.indexOf(this.startPositionName) : 0;\n    this.endPosition = (this.endPositionName && this.positionNames &&\n      this.positionNames.includes(this.endPositionName)) ?\n      this.positionNames.indexOf(this.endPositionName) : (maxLength - 1);\n  }\n\n  public override onPropertyChanged(property: DG.Property): void {\n    // console.debug(`WebLogo.onPropertyChanged( ${property.name} = '' })`);\n    super.onPropertyChanged(property);\n\n    switch (property.name) {\n    case 'considerNullSequences':\n      this.render(true);\n      break;\n    case 'sequenceColumnName':\n      this.updateSeqCol();\n      break;\n    case 'startPositionName':\n      this.updateSeqCol();\n      break;\n    case 'endPositionName':\n      this.updateSeqCol();\n      break;\n    case 'positionWidth':\n      this._positionWidth = this.positionWidth;\n      this.render(true);\n      break;\n    case 'minHeight':\n      this.render(true);\n      break;\n    case 'maxHeight':\n      this.render(true);\n      break;\n    case 'fixWidth':\n      this.render(true);\n      break;\n    case 'verticalAlignment':\n      this.render(true);\n      break;\n    case 'horizontalAlignment':\n      this.render(true);\n      break;\n    case 'fitArea':\n      this.render(true);\n      break;\n    case 'shrinkEmptyTail':\n      this.updatePositions();\n      this.render(true);\n      break;\n    case 'skipEmptyPositions':\n      this.updatePositions();\n      this.render(true);\n      break;\n    case 'positionMargin':\n      this.render(true);\n      break;\n    case 'positionMarginState':\n      this.render(true);\n      break;\n    case 'positionHeight':\n      this.render(true);\n      break;\n    }\n  }\n\n  public override async onTableAttached() {\n    const dataFrameTxt: string = this.dataFrame ? 'data' : 'null';\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n    super.onTableAttached();\n\n    this.updateSeqCol();\n\n    if (this.dataFrame !== void 0) {\n      this.subs.push(this.dataFrame.selection.onChanged.subscribe((_) => this.render()));\n      this.subs.push(this.dataFrame.filter.onChanged.subscribe((_) => {\n        this.updatePositions();\n        this.render();\n      }));\n    }\n\n    await this.init();\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n  }\n\n  public override async detach() {\n    const dataFrameTxt = `${this.dataFrame ? 'data' : 'null'}`;\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n    super.detach();\n\n    this.viewSubs.forEach((sub) => sub.unsubscribe());\n    this.host!.remove();\n    this.canvas = undefined;\n    this.msgHost = undefined;\n    this.host = undefined;\n\n    this.initialized = false;\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n  }\n\n  protected _nullSequence(fillerResidue = 'X'): string {\n    if (this.considerNullSequences)\n      return new Array(this.Length).fill(fillerResidue).join('');\n\n    return '';\n  }\n\n  protected removeWhere(array: Array<any>, predicate: (T: any) => boolean): Array<any> {\n    let length = array.length;\n    let updateIterator = 0;\n    for (let deleteIterator = 0; deleteIterator < length; deleteIterator++) {\n      if (!predicate(array[deleteIterator])) {\n        array[updateIterator] = array[deleteIterator];\n        updateIterator++;\n      }\n    }\n    array.length = updateIterator;\n    return array;\n  }\n\n\n  protected _removeEmptyPositions() {\n    if (this.skipEmptyPositions) {\n      this.removeWhere(this.positions, item => item?.freq['-']?.count === item.rowCount);\n    }\n  }\n\n  protected _calculate(r: number) {\n    if (!this.canvas || !this.host || !this.seqCol || !this.dataFrame)\n      return;\n    this.unitsHandler = new UnitsHandler(this.seqCol);\n\n\n    this.calcSize();\n\n    this.positions = new Array(this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0);\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const posName: string = this.positionNames[this.startPosition + jPos];\n      this.positions[jPos] = new PositionInfo(posName);\n    }\n\n    // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)\n    const indices = this.dataFrame.filter.getSelectedIndexes();\n    // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :\n    //   this.dataFrame.filter.getSelectedIndexes();\n\n    this.rowsMasked = indices.length;\n    this.rowsNull = 0;\n\n    for (const i of indices) {\n      let s: string = <string>(this.seqCol.get(i));\n\n      if (!s) {\n        s = this._nullSequence();\n        ++this.rowsNull;\n      }\n\n      const seqM: string[] = this.splitter!(s);\n      for (let jPos = 0; jPos < this.Length; jPos++) {\n        const pmInfo = this.positions[jPos].freq;\n        const m: string = seqM[this.startPosition + jPos] || '-';\n        if (!(m in pmInfo))\n          pmInfo[m] = new PositionMonomerInfo();\n        pmInfo[m].count++;\n      }\n    }\n\n    //#region Polish freq counts\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      // delete this.positions[jPos].freq['-'];\n\n      this.positions[jPos].rowCount = 0;\n      for (const m in this.positions[jPos].freq)\n        this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;\n      if (this.positionHeight == PositionHeight.Entropy) {\n        this.positions[jPos].sumForHeightCalc = 0;\n        for (const m in this.positions[jPos].freq) {\n          const pn = this.positions[jPos].freq[m].count / this.positions[jPos].rowCount;\n          this.positions[jPos].sumForHeightCalc += -pn * Math.log2(pn);\n        }\n      }\n    }\n    //#endregion\n    this._removeEmptyPositions();\n\n    const absoluteMaxHeight = this.canvas.height - this.axisHeight * r;\n    // console.debug(`WebLogo<${this.viewerId}>._calculate() maxHeight=${maxHeight}.`);\n\n    //#region Calculate screen\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const freq: { [c: string]: PositionMonomerInfo } = this.positions[jPos].freq;\n      const rowCount = this.positions[jPos].rowCount;\n      const alphabetSize = this.getAlphabetSize();\n      if ((this.positionHeight == PositionHeight.Entropy) && (alphabetSize == null)) {\n        grok.shell.error('WebLogo: alphabet is undefined.');\n      }\n\n      const maxHeight = (this.positionHeight == PositionHeight.Entropy) ? (absoluteMaxHeight * (Math.log2(alphabetSize) - (this.positions[jPos].sumForHeightCalc)) / Math.log2(alphabetSize)) : absoluteMaxHeight;\n\n      let y: number = this.axisHeight * r + (absoluteMaxHeight - maxHeight - 1);\n\n      const entries = Object.entries(freq).sort((a, b) => {\n        if (a[0] !== '-' && b[0] !== '-')\n          return b[1].count - a[1].count;\n        else if (a[0] === '-' && b[0] === '-')\n          return 0;\n        else if (a[0] === '-')\n          return -1;\n        else /* (b[0] === '-') */\n          return +1;\n      });\n      for (const entry of entries) {\n        const pmInfo: PositionMonomerInfo = entry[1];\n        // const m: string = entry[0];\n        const h: number = maxHeight * pmInfo.count / rowCount;\n\n        pmInfo.bounds = new DG.Rect(jPos * this.positionWidthWithMargin, y, this._positionWidth, h);\n        y += h;\n      }\n    }\n    //#endregion\n  }\n\n  // reflect changes made to filter/selection\n  render(recalc = true) {\n    if (this.msgHost) {\n      if (this.seqCol && !this.cp) {\n        this.msgHost!.innerText = `Unknown palette (column semType: '${this.seqCol.semType}').`;\n        this.msgHost!.style.display = '';\n      } else {\n        this.msgHost!.style.display = 'none';\n      }\n    }\n\n    if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp)\n      return;\n\n    const g = this.canvas.getContext('2d');\n    if (!g) return;\n\n    const r = window.devicePixelRatio;\n\n    if (recalc)\n      this._calculate(r);\n\n    g.resetTransform();\n    g.fillStyle = 'white';\n    g.fillRect(0, 0, this.canvas.width, this.canvas.height);\n    g.textBaseline = this.textBaseline;\n\n    // Prevents division by zero on Length = 0\n    if (this.startPosition === -1 || this.endPosition === -1)\n      return;\n\n    //#region Plot positionNames\n    const positionFontSize = 10 * r;\n    g.resetTransform();\n    g.fillStyle = 'black';\n    g.textAlign = 'center';\n    g.font = `${positionFontSize.toFixed(1)}px Roboto, Roboto Local, sans-serif`;\n    const posNameMaxWidth = Math.max(...this.positions.map((pos) => g.measureText(pos.name).width));\n    const hScale = posNameMaxWidth < (this._positionWidth - 2) ? 1 : (this._positionWidth - 2) / posNameMaxWidth;\n\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const pos: PositionInfo = this.positions[jPos];\n      g.resetTransform();\n      g.setTransform(\n        hScale, 0, 0, 1,\n        jPos * this.positionWidthWithMargin + this._positionWidth / 2, 0);\n      g.fillText(pos.name, 0, 0);\n    }\n    //#endregion Plot positionNames\n\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {\n        if (monomer !== '-') {\n          const monomerTxt = WebLogo.monomerToText(monomer);\n          const b = pmInfo.bounds;\n\n          const fontStyle = '16px Roboto, Roboto Local, sans-serif';\n          // Hacks to scale uppercase characters to target rectangle\n          const uppercaseLetterAscent = 0.25;\n          const uppercaseLetterHeight = 12.2;\n\n          g.resetTransform();\n          g.strokeStyle = 'lightgray';\n          g.lineWidth = 1;\n          g.rect(b.left, b.top, b.width, b.height);\n          g.fillStyle = this.cp.get(monomer) ?? this.cp.get('other');\n          g.textAlign = 'left';\n          g.font = fontStyle;\n          //g.fillRect(b.left, b.top, b.width, b.height);\n          const mTm: TextMetrics = g.measureText(monomerTxt);\n\n          // if (mM.actualBoundingBoxAscent != 0)\n          //   console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);\n\n          g.setTransform(\n            b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight,\n            b.left, b.top);\n          g.fillText(monomerTxt, 0, -uppercaseLetterAscent);\n        }\n      }\n    }\n  }\n\n  private calcSize() {\n    if (!this.canvas || !this.host)\n      return;\n\n    const r: number = window.devicePixelRatio;\n\n    let width: number = this.Length * this.positionWidthWithMargin / r;\n    let height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));\n\n    if (this.fitArea) {\n      const xScale: number = this.root.clientHeight / height;\n      const yScale: number = this.root.clientWidth / width;\n      const scale = Math.max(1, Math.min(xScale, yScale));\n      width = width * scale;\n      height = height * scale;\n      this._positionWidth = this.positionWidth * scale;\n    }\n\n    this.canvas.width = width * r;\n    this.canvas.style.width = `${width}px`;\n\n    // const canvasHeight: number = width > this.root.clientWidth ? height - 8 : height;\n    this.host.style.setProperty('height', `${height}px`);\n    const canvasHeight: number = this.host.clientHeight;\n    this.canvas.height = canvasHeight * r;\n    this.canvas.style.setProperty('height', `${canvasHeight}px`);\n\n    // Adjust host and root width\n    if (this.fixWidth) {\n      // full width for canvas host and root\n      this.root.style.width = this.host.style.width = `${width}px`;\n      this.root.style.height = `${height}px`;\n      this.host.style.setProperty('overflow', 'hidden', 'important');\n    } else {\n      // allow scroll canvas in root\n      this.root.style.width = this.host.style.width = '100%';\n      this.host.style.overflowX = 'auto!important';\n      this.host.style.setProperty('overflow', null);\n\n      this.host.style.setProperty('text-align', this.horizontalAlignment);\n\n      // vertical alignment\n      let hostTopMargin = 0;\n      switch (this.verticalAlignment) {\n      case 'top':\n        hostTopMargin = 0;\n        break;\n      case 'middle':\n        hostTopMargin = Math.max(0, (this.root.clientHeight - height) / 2);\n        break;\n      case 'bottom':\n        hostTopMargin = Math.max(0, this.root.clientHeight - height);\n        break;\n      }\n      this.host.style.setProperty('margin-top', `${hostTopMargin}px`, 'important');\n\n      if (this.root.clientHeight < height) {\n        this.host.style.setProperty('height', `${this.root.clientHeight}px`);\n        this.host.style.setProperty('overflow-y', null);\n      } else {\n        this.host.style.setProperty('overflow-y', 'hidden', 'important');\n      }\n    }\n\n    // console.debug(\n    //   `this.root.style.height = ${this.root.style.height}\\n` +\n    //   `this.root.clientHeight = ${this.root.clientHeight}\\n` +\n    //   `this.host.style.height = ${this.host.style.height}\\n` +\n    //   `this.host.clientHeight = ${this.host.clientHeight}\\n` +\n    //   '\\n' +\n    //   `this.canvas.height       = ${this.canvas.height}\\n` +\n    //   `this.canvas.style.height = ${this.canvas.style.height}`);\n  }\n\n  /**\n   * @param {DG.Column} seqCol Column to look for a palette\n   * @param {number}  minLength minimum length of sequence to detect palette (empty strings are allowed)\n   * @return {SeqPalette} Palette corresponding to the alphabet of the sequences in the column\n   */\n  public static pickUpPalette(seqCol: DG.Column, minLength: number = 5): SeqPalette {\n    let res: SeqPalette | null = null;\n    switch (seqCol.semType) {\n    case Aminoacids.SemTypeMultipleAlignment:\n      res = AminoacidsPalettes.GrokGroups;\n      break;\n    case Nucleotides.SemTypeMultipleAlignment:\n      res = NucleotidesPalettes.Chromatogram;\n      break;\n    }\n    const stats: SeqColStats = WebLogo.getStats(seqCol, minLength, WebLogo.splitterAsFasta);\n\n    const alphabetCandidates: [Set<string>, SeqPalette][] = [\n      [new Set(Object.keys(Nucleotides.Names)), NucleotidesPalettes.Chromatogram],\n      [new Set(Object.keys(Aminoacids.Names)), AminoacidsPalettes.GrokGroups],\n    ];\n    // Calculate likelihoods for alphabet_candidates\n    const alphabetCandidatesSim: number[] = alphabetCandidates\n      .map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[0]));\n    const maxCos = Math.max(...alphabetCandidatesSim);\n    if (maxCos > 0.55)\n      res = alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][1];\n    else\n      res = UnknownSeqPalettes.Color;\n\n    return res;\n  }\n\n  public getAlphabetSize(): number {\n    return this.unitsHandler?.getAlphabetSize() ?? 0;\n  }\n\n  /** Stats of sequences with specified splitter func, returns { freq, sameLength }.\n   * @param {DG.Column} seqCol\n   * @param {number} minLength\n   * @param {SplitterFunc} splitter\n   * @return { SeqColStats }, sameLength: boolean } stats of column sequences\n   */\n  static getStats(seqCol: DG.Column, minLength: number, splitter: SplitterFunc): SeqColStats {\n    const freq: { [m: string]: number } = {};\n    let sameLength = true;\n    let firstLength = null;\n\n    for (const seq of seqCol.categories) {\n      const mSeq = splitter(seq);\n\n      if (firstLength == null)\n        firstLength = mSeq.length;\n      else if (mSeq.length !== firstLength)\n        sameLength = false;\n\n      if (mSeq.length > minLength) {\n        for (const m of mSeq) {\n          if (!(m in freq))\n            freq[m] = 0;\n          freq[m] += 1;\n        }\n      }\n    }\n    return {freq: freq, sameLength: sameLength};\n  }\n\n  public static getAlphabetSimilarity(freq: MonomerFreqs, alphabet: Set<string>, gapSymbol: string = '-'): number {\n    const keys = new Set<string>([...new Set(Object.keys(freq)), ...alphabet]);\n    keys.delete(gapSymbol);\n\n    const freqA: number[] = [];\n    const alphabetA: number[] = [];\n    for (const m of keys) {\n      freqA.push(m in freq ? freq[m] : 0);\n      alphabetA.push(alphabet.has(m) ? 1 : 0);\n    }\n    /* There were a few ideas: chi-squared, pearson correlation (variance?), scalar product */\n    const freqV: Vector = new Vector(freqA);\n    const alphabetV: Vector = new Vector(alphabetA);\n    return vectorDotProduct(freqV, alphabetV) / (vectorLength(freqV) * vectorLength(alphabetV));\n  }\n\n  // /** First try to find column with semType 'alignedSequence'.\n  //  * Next look for column with data alphabet corresponding to any of the known palettes.\n  //  * @param {DG.DataFrame} dataFrame\n  //  * @return {DG.Column} The column we were looking for or null\n  //  */\n  // private pickUpSeqCol(dataFrame: DG.DataFrame): DG.Column | null {\n  //   let res: DG.Column | null = dataFrame.columns.bySemType('alignedSequence');\n  //   if (res == null) {\n  //     for (const col of dataFrame.columns) {\n  //       const cp = WebLogo.pickUpPalette(col as DG.Column, 5);\n  //       if (cp !== null && !(cp instanceof UnknownSeqPalette)) {\n  //         res = col;\n  //         break;\n  //       }\n  //     }\n  //   }\n  //   return res;\n  // }\n\n  public static pickUpSeqCol2(df: DG.DataFrame): DG.Column | null {\n    const semTypeColList = df.columns.bySemTypeAll(DG.SEMTYPE.MACROMOLECULE);\n    let resCol: DG.Column | null = semTypeColList.find((col) => {\n      const units = col.getTag(DG.TAGS.UNITS);\n      return units ? units.indexOf('MSA') !== -1 : false;\n    }) ?? null;\n    if (!resCol && semTypeColList.length > 0)\n      resCol = semTypeColList[0];\n    return resCol;\n  }\n\n  private static monomerRe = /\\[(\\w+)\\]|(\\w)|(-)/g;\n\n  /** Split sequence for single character monomers, square brackets multichar monomer names or gap symbol.\n   * @param {any} seq object with sequence\n   * @return {string[]} array of monomers\n   */\n  public static splitterAsFasta(seq: any): string[] {\n    const res: string[] = wu<RegExpMatchArray>(seq.toString().matchAll(WebLogo.monomerRe))\n      .map((ma: RegExpMatchArray) => {\n        let mRes: string;\n        const m: string = ma[0];\n        if (m.length > 1) {\n          if (m in WebLogo.aaSynonyms) {\n            mRes = WebLogo.aaSynonyms[m];\n          } else {\n            mRes = '';\n            console.debug(`Long monomer '${m}' has not a short synonym.`);\n          }\n        } else {\n          mRes = m;\n        }\n        return mRes;\n      }).toArray();\n\n    return res;\n  }\n\n  private static helmRe = /(PEPTIDE1|DNA1|RNA1)\\{([^}]+)}/g;\n  private static helmPp1Re = /\\[([^\\[\\]]+)]/g;\n\n  /** Splits Helm string to monomers, but does not replace monomer names to other notation (e.g. for RNA).\n   * Only for linear polymers, does not split RNA for ribose and phosphate monomers.\n   * @param {string} seq Source string of HELM notation\n   * @return {string[]}\n   */\n  public static splitterAsHelm(seq: any): string[] {\n    WebLogo.helmRe.lastIndex = 0;\n    const ea: RegExpExecArray | null = WebLogo.helmRe.exec(seq.toString());\n    const inSeq: string | null = ea ? ea[2] : null;\n\n    const mmPostProcess = (mm: string): string => {\n      WebLogo.helmPp1Re.lastIndex = 0;\n      const pp1M = WebLogo.helmPp1Re.exec(mm);\n      if (pp1M && pp1M.length >= 2)\n        return pp1M[1];\n      else\n        return mm;\n    };\n\n    const mmList: string[] = inSeq ? inSeq.split('.') : [];\n    const mmListRes: string[] = mmList.map(mmPostProcess);\n    return mmListRes;\n  }\n\n  /** Gets method to split sequence by separator\n   * @param {string} separator\n   * @param limit\n   * @return {SplitterFunc}\n   */\n  public static getSplitterWithSeparator(separator: string, limit: number | undefined = undefined): SplitterFunc {\n    return (seq: string) => {\n      return seq.split(separator, limit);\n    };\n  }\n\n  /** Get splitter method to split sequences to monomers\n   * @param {string} units\n   * @param {string} separator\n   * @param limit\n   * @return {SplitterFunc}\n   */\n  public static getSplitter(units: string, separator: string, limit: number | undefined = undefined): SplitterFunc {\n    if (units.toLowerCase().startsWith(NOTATION.FASTA))\n      return WebLogo.splitterAsFasta;\n    else if (units.toLowerCase().startsWith(NOTATION.SEPARATOR))\n      return WebLogo.getSplitterWithSeparator(separator, limit);\n    else if (units.toLowerCase().startsWith(NOTATION.HELM))\n      return WebLogo.splitterAsHelm;\n    else\n      throw new Error(`Unexpected units ${units} .`);\n\n    // TODO: Splitter for HELM\n  }\n\n  public static getSplitterForColumn(col: DG.Column): SplitterFunc {\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n      throw new Error(`Get splitter for semType \"${DG.SEMTYPE.MACROMOLECULE}\" only.`);\n\n    const units = col.getTag(DG.TAGS.UNITS);\n    const separator = col.getTag(UnitsHandler.TAGS.separator);\n    return WebLogo.getSplitter(units, separator);\n  }\n\n  /** Only some of the synonyms. These were obtained from the clustered oligopeptide dataset. */\n  private static aaSynonyms: { [name: string]: string } = {\n    '[MeNle]': 'L', // Nle - norleucine\n    '[MeA]': 'A',\n    '[MeG]': 'G',\n    '[MeF]': 'F',\n  };\n\n  private static longMonomerPartRe = /(\\w+)/g;\n\n  /* Shortens monomer representation text for monomers with long names.\n  **/\n  public static monomerToText(src: any): string {\n    const srcTxt: string = src.toString();\n    if (srcTxt.length <= 5) {\n      return srcTxt;\n    } else {\n      const parts: string[] = wu<RegExpMatchArray>(src.toString().matchAll(WebLogo.longMonomerPartRe))\n        .map((ma: RegExpMatchArray) => {\n          const mRes: string = ma[0];\n          return mRes;\n        }).toArray();\n\n      if (parts.length == 0) {\n        return ' ';\n      } else {\n        const part0: string = parts[0];\n        const resTxt: string = part0.length < 6 ? `${part0}…` : `${part0.substring(0, 5)}…`;\n        return resTxt;\n      }\n    }\n  }\n\n  public static monomerToShort(amino: string, maxLengthOfMonomer: number): string {\n    return amino.length <= maxLengthOfMonomer ? amino : amino.substring(0, maxLengthOfMonomer) + '…';\n  }\n}\n"]}","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport { normalize } from '@datagrok-libraries/utils/src/vector-operations';\nimport { createDimensinalityReducingWorker } from './workers/dimensionality-reducing-worker-creator';\nexport function reduceDimensinalityWithNormalization(dataCol, methodName, similarityMetric, options) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const dimensionalityReduceRes = yield createDimensinalityReducingWorker({ data: dataCol, metric: similarityMetric }, methodName, options);\n        dimensionalityReduceRes.embedding = dimensionalityReduceRes.embedding.map(it => normalize(it));\n        return dimensionalityReduceRes;\n    });\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxdWVuY2Utc3BhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzZXF1ZW5jZS1zcGFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFFQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0saURBQWlELENBQUM7QUFHNUUsT0FBTyxFQUFFLGlDQUFpQyxFQUErQixNQUFNLGtEQUFrRCxDQUFDO0FBR2xJLE1BQU0sVUFBZ0Isb0NBQW9DLENBQ3RELE9BQXFDLEVBQ3JDLFVBQWtCLEVBQ2xCLGdCQUE2RCxFQUM3RCxPQUFhOztRQUViLE1BQU0sdUJBQXVCLEdBQ3pCLE1BQU0saUNBQWlDLENBQ25DLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsZ0JBQWdCLEVBQWdCLEVBQ3pELFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUU3Qix1QkFBdUIsQ0FBQyxTQUFTLEdBQUcsdUJBQXVCLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQy9GLE9BQU8sdUJBQXVCLENBQUM7SUFDbkMsQ0FBQztDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJpdEFycmF5IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL2JpdC1hcnJheSc7XG5pbXBvcnQgeyBNYXRyaXgsIFZlY3RvciB9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3R5cGUtZGVjbGFyYXRpb25zJztcbmltcG9ydCB7IG5vcm1hbGl6ZSB9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3ZlY3Rvci1vcGVyYXRpb25zJztcbmltcG9ydCAqIGFzIERHIGZyb20gJ2RhdGFncm9rLWFwaS9kZyc7XG5pbXBvcnQgeyBCaXRBcnJheU1ldHJpY3MsIFN0cmluZ01ldHJpY3MsIFZhbGlkVHlwZXMsIFZlY3Rvck1ldHJpY3MgfSBmcm9tICcuL3R5cGVkLW1ldHJpY3MnO1xuaW1wb3J0IHsgY3JlYXRlRGltZW5zaW5hbGl0eVJlZHVjaW5nV29ya2VyLCBJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQgfSBmcm9tICcuL3dvcmtlcnMvZGltZW5zaW9uYWxpdHktcmVkdWNpbmctd29ya2VyLWNyZWF0b3InO1xuXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiByZWR1Y2VEaW1lbnNpbmFsaXR5V2l0aE5vcm1hbGl6YXRpb24oXG4gICAgZGF0YUNvbDogQml0QXJyYXlbXXxWZWN0b3JbXXxzdHJpbmdbXSwgXG4gICAgbWV0aG9kTmFtZTogc3RyaW5nLCBcbiAgICBzaW1pbGFyaXR5TWV0cmljOiBCaXRBcnJheU1ldHJpY3N8VmVjdG9yTWV0cmljc3xTdHJpbmdNZXRyaWNzLFxuICAgIG9wdGlvbnM/OiBhbnkpOiBQcm9taXNlPElSZWR1Y2VEaW1lbnNpb25hbGl0eVJlc3VsdD4ge1xuXG4gICAgY29uc3QgZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXM6IElSZWR1Y2VEaW1lbnNpb25hbGl0eVJlc3VsdCA9XG4gICAgICAgIGF3YWl0IGNyZWF0ZURpbWVuc2luYWxpdHlSZWR1Y2luZ1dvcmtlcihcbiAgICAgICAgICAgIHsgZGF0YTogZGF0YUNvbCwgbWV0cmljOiBzaW1pbGFyaXR5TWV0cmljIH0gYXMgVmFsaWRUeXBlcyxcbiAgICAgICAgICAgIG1ldGhvZE5hbWUsIG9wdGlvbnMpO1xuXG4gICAgZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXMuZW1iZWRkaW5nID0gZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXMuZW1iZWRkaW5nLm1hcChpdCA9PiBub3JtYWxpemUoaXQpKTtcbiAgICByZXR1cm4gZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXM7XG59Il19","import * as fl from 'fastest-levenshtein';\nimport { jaroWinkler } from 'jaro-winkler-typescript';\nimport { distanceMetrics } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { calculateEuclideanDistance } from '@datagrok-libraries/utils/src/vector-operations';\nexport const AvailableMetrics = {\n    'Vector': {\n        'EuclideanDistance': calculateEuclideanDistance,\n    },\n    'String': {\n        'Levenshtein': fl.distance,\n        'Jaro-Winkler': jaroWinkler,\n    },\n    'BitArray': {\n        'Tanimoto': distanceMetrics['Tanimoto'],\n        'Dice': distanceMetrics['Dice'],\n        'Asymmetric': distanceMetrics['Asymmetric'],\n        'Braun-Blanquet': distanceMetrics['Braun-Blanquet'],\n        'Cosine': distanceMetrics['Cosine'],\n        'Kulczynski': distanceMetrics['Kulczynski'],\n        'Mc-Connaughey': distanceMetrics['Mc-Connaughey'],\n        'Rogot-Goldberg': distanceMetrics['Rogot-Goldberg'],\n        'Russel': distanceMetrics['Russel'],\n        'Sokal': distanceMetrics['Sokal'],\n    },\n};\nexport const MetricToDataType = Object.keys(AvailableMetrics)\n    .reduce((ret, key) => {\n    for (const val of Object.keys(AvailableMetrics[key])) {\n        ret[val] = key;\n    }\n    return ret;\n}, {});\nexport function isStringMetric(name) {\n    return MetricToDataType[name] == 'String';\n}\nexport function isBitArrayMetric(name) {\n    return MetricToDataType[name] == 'BitArray';\n}\nexport function isVectorMetric(name) {\n    return MetricToDataType[name] == 'Vector';\n}\n/** Unified class implementing different string measures. */\nexport class Measure {\n    /**\n     * Creates an instance of Measure with .\n     * @param {string} method Method to calculate distance between strings.\n     * @memberof Measurer\n     */\n    constructor(method) {\n        this.method = method;\n        this.dataType = MetricToDataType[method];\n    }\n    /**\n     * Returns custom string distance function specified.\n     * @return {DistanceMetric} Callback of the measure chosen.\n     * @memberof Measurer\n     */\n    getMeasure() {\n        const dict = AvailableMetrics;\n        return dict[this.dataType][this.method];\n    }\n    /**\n     * Returns custom string distance by the given data type.\n     * @param {AvailableDataTypes} dataType Metric's data type\n     * @return {string[]} Metric names which expects the given data type\n     * @memberof Measurer\n     */\n    static getMetricByDataType(dataType) {\n        return Object.keys(AvailableMetrics[dataType]);\n    }\n    /** Returns metric names available.\n     * @memberof Measurer\n    */\n    static get availableMeasures() {\n        return Object.keys(AvailableMetrics);\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWQtbWV0cmljcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInR5cGVkLW1ldHJpY3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUMxQyxPQUFPLEVBQUMsV0FBVyxFQUFDLE1BQU0seUJBQXlCLENBQUM7QUFFcEQsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLGtEQUFrRCxDQUFDO0FBQ2pGLE9BQU8sRUFBQywwQkFBMEIsRUFBQyxNQUFNLGlEQUFpRCxDQUFDO0FBSTNGLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHO0lBQzlCLFFBQVEsRUFBRTtRQUNSLG1CQUFtQixFQUFFLDBCQUEwQjtLQUNoRDtJQUNELFFBQVEsRUFBRTtRQUNSLGFBQWEsRUFBRSxFQUFFLENBQUMsUUFBUTtRQUMxQixjQUFjLEVBQUUsV0FBVztLQUM1QjtJQUNELFVBQVUsRUFBRTtRQUNWLFVBQVUsRUFBRSxlQUFlLENBQUMsVUFBVSxDQUFDO1FBQ3ZDLE1BQU0sRUFBRSxlQUFlLENBQUMsTUFBTSxDQUFDO1FBQy9CLFlBQVksRUFBRSxlQUFlLENBQUMsWUFBWSxDQUFDO1FBQzNDLGdCQUFnQixFQUFFLGVBQWUsQ0FBQyxnQkFBZ0IsQ0FBQztRQUNuRCxRQUFRLEVBQUUsZUFBZSxDQUFDLFFBQVEsQ0FBQztRQUNuQyxZQUFZLEVBQUUsZUFBZSxDQUFDLFlBQVksQ0FBQztRQUMzQyxlQUFlLEVBQUUsZUFBZSxDQUFDLGVBQWUsQ0FBQztRQUNqRCxnQkFBZ0IsRUFBRSxlQUFlLENBQUMsZ0JBQWdCLENBQUM7UUFDbkQsUUFBUSxFQUFFLGVBQWUsQ0FBQyxRQUFRLENBQUM7UUFDbkMsT0FBTyxFQUFFLGVBQWUsQ0FBQyxPQUFPLENBQUM7S0FDbEM7Q0FDRixDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sZ0JBQWdCLEdBQXFCLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUM7S0FDNUUsTUFBTSxDQUFDLENBQUMsR0FBcUIsRUFBRSxHQUFHLEVBQUUsRUFBRTtJQUNyQyxLQUFLLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsR0FBeUIsQ0FBQyxDQUFDLEVBQUU7UUFDMUUsR0FBRyxDQUFDLEdBQXlCLENBQUMsR0FBRyxHQUFHLENBQUM7S0FDdEM7SUFDRCxPQUFPLEdBQUcsQ0FBQztBQUNiLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztBQVdULE1BQU0sVUFBVSxjQUFjLENBQUMsSUFBa0I7SUFDL0MsT0FBTyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxRQUFRLENBQUM7QUFDNUMsQ0FBQztBQUVELE1BQU0sVUFBVSxnQkFBZ0IsQ0FBQyxJQUFrQjtJQUNqRCxPQUFPLGdCQUFnQixDQUFDLElBQUksQ0FBQyxJQUFJLFVBQVUsQ0FBQztBQUM5QyxDQUFDO0FBRUQsTUFBTSxVQUFVLGNBQWMsQ0FBQyxJQUFrQjtJQUMvQyxPQUFPLGdCQUFnQixDQUFDLElBQUksQ0FBQyxJQUFJLFFBQVEsQ0FBQztBQUM1QyxDQUFDO0FBRUQsNERBQTREO0FBQzVELE1BQU0sT0FBTyxPQUFPO0lBSWxCOzs7O09BSUc7SUFDSCxZQUFZLE1BQW9CO1FBQzlCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO1FBQ3JCLElBQUksQ0FBQyxRQUFRLEdBQUcsZ0JBQWdCLENBQUMsTUFBTSxDQUF1QixDQUFDO0lBQ2pFLENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksVUFBVTtRQUNmLE1BQU0sSUFBSSxHQUFzRCxnQkFBZ0IsQ0FBQztRQUNqRixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzFDLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxRQUE0QjtRQUM1RCxPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztJQUNqRCxDQUFDO0lBRUQ7O01BRUU7SUFDRixNQUFNLEtBQUssaUJBQWlCO1FBQzFCLE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGZsIGZyb20gJ2Zhc3Rlc3QtbGV2ZW5zaHRlaW4nO1xuaW1wb3J0IHtqYXJvV2lua2xlcn0gZnJvbSAnamFyby13aW5rbGVyLXR5cGVzY3JpcHQnO1xuaW1wb3J0IHtEaXN0YW5jZU1ldHJpY30gZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvdHlwZS1kZWNsYXJhdGlvbnMnO1xuaW1wb3J0IHtkaXN0YW5jZU1ldHJpY3N9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3NpbWlsYXJpdHktbWV0cmljcyc7XG5pbXBvcnQge2NhbGN1bGF0ZUV1Y2xpZGVhbkRpc3RhbmNlfSBmcm9tICdAZGF0YWdyb2stbGlicmFyaWVzL3V0aWxzL3NyYy92ZWN0b3Itb3BlcmF0aW9ucyc7XG5pbXBvcnQgQml0QXJyYXkgZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvYml0LWFycmF5JztcbmltcG9ydCB7VmVjdG9yLCBTdHJpbmdEaWN0aW9uYXJ5fSBmcm9tICdAZGF0YWdyb2stbGlicmFyaWVzL3V0aWxzL3NyYy90eXBlLWRlY2xhcmF0aW9ucyc7XG5cbmV4cG9ydCBjb25zdCBBdmFpbGFibGVNZXRyaWNzID0ge1xuICAnVmVjdG9yJzoge1xuICAgICdFdWNsaWRlYW5EaXN0YW5jZSc6IGNhbGN1bGF0ZUV1Y2xpZGVhbkRpc3RhbmNlLFxuICB9LFxuICAnU3RyaW5nJzoge1xuICAgICdMZXZlbnNodGVpbic6IGZsLmRpc3RhbmNlLFxuICAgICdKYXJvLVdpbmtsZXInOiBqYXJvV2lua2xlcixcbiAgfSxcbiAgJ0JpdEFycmF5Jzoge1xuICAgICdUYW5pbW90byc6IGRpc3RhbmNlTWV0cmljc1snVGFuaW1vdG8nXSxcbiAgICAnRGljZSc6IGRpc3RhbmNlTWV0cmljc1snRGljZSddLFxuICAgICdBc3ltbWV0cmljJzogZGlzdGFuY2VNZXRyaWNzWydBc3ltbWV0cmljJ10sXG4gICAgJ0JyYXVuLUJsYW5xdWV0JzogZGlzdGFuY2VNZXRyaWNzWydCcmF1bi1CbGFucXVldCddLFxuICAgICdDb3NpbmUnOiBkaXN0YW5jZU1ldHJpY3NbJ0Nvc2luZSddLFxuICAgICdLdWxjenluc2tpJzogZGlzdGFuY2VNZXRyaWNzWydLdWxjenluc2tpJ10sXG4gICAgJ01jLUNvbm5hdWdoZXknOiBkaXN0YW5jZU1ldHJpY3NbJ01jLUNvbm5hdWdoZXknXSxcbiAgICAnUm9nb3QtR29sZGJlcmcnOiBkaXN0YW5jZU1ldHJpY3NbJ1JvZ290LUdvbGRiZXJnJ10sXG4gICAgJ1J1c3NlbCc6IGRpc3RhbmNlTWV0cmljc1snUnVzc2VsJ10sXG4gICAgJ1Nva2FsJzogZGlzdGFuY2VNZXRyaWNzWydTb2thbCddLFxuICB9LFxufTtcblxuZXhwb3J0IGNvbnN0IE1ldHJpY1RvRGF0YVR5cGU6IFN0cmluZ0RpY3Rpb25hcnkgPSBPYmplY3Qua2V5cyhBdmFpbGFibGVNZXRyaWNzKVxuICAucmVkdWNlKChyZXQ6IFN0cmluZ0RpY3Rpb25hcnksIGtleSkgPT4ge1xuICAgIGZvciAoY29uc3QgdmFsIG9mIE9iamVjdC5rZXlzKEF2YWlsYWJsZU1ldHJpY3Nba2V5IGFzIEF2YWlsYWJsZURhdGFUeXBlc10pKSB7XG4gICAgICByZXRbdmFsIGFzIEF2YWlsYWJsZURhdGFUeXBlc10gPSBrZXk7XG4gICAgfVxuICAgIHJldHVybiByZXQ7XG4gIH0sIHt9KTtcblxuZXhwb3J0IHR5cGUgQXZhaWxhYmxlRGF0YVR5cGVzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3M7XG5leHBvcnQgdHlwZSBTdHJpbmdNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ1N0cmluZyddO1xuZXhwb3J0IHR5cGUgQml0QXJyYXlNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ0JpdEFycmF5J107XG5leHBvcnQgdHlwZSBWZWN0b3JNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ1ZlY3RvciddO1xuZXhwb3J0IHR5cGUgS25vd25NZXRyaWNzID0gU3RyaW5nTWV0cmljcyB8IEJpdEFycmF5TWV0cmljcyB8IFZlY3Rvck1ldHJpY3M7XG5cbmV4cG9ydCB0eXBlIFZhbGlkVHlwZXMgPSB7ZGF0YTogc3RyaW5nW10sIG1ldHJpYzogU3RyaW5nTWV0cmljc30gfCB7ZGF0YTogVmVjdG9yW10sIG1ldHJpYzogVmVjdG9yTWV0cmljc30gfFxuICAgICAgICAgICAgICAgICAgICAgICAgIHtkYXRhOiBCaXRBcnJheVtdLCBtZXRyaWM6IEJpdEFycmF5TWV0cmljc307XG5cbmV4cG9ydCBmdW5jdGlvbiBpc1N0cmluZ01ldHJpYyhuYW1lOiBLbm93bk1ldHJpY3MpIHtcbiAgcmV0dXJuIE1ldHJpY1RvRGF0YVR5cGVbbmFtZV0gPT0gJ1N0cmluZyc7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBpc0JpdEFycmF5TWV0cmljKG5hbWU6IEtub3duTWV0cmljcykge1xuICByZXR1cm4gTWV0cmljVG9EYXRhVHlwZVtuYW1lXSA9PSAnQml0QXJyYXknO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNWZWN0b3JNZXRyaWMobmFtZTogS25vd25NZXRyaWNzKSB7XG4gIHJldHVybiBNZXRyaWNUb0RhdGFUeXBlW25hbWVdID09ICdWZWN0b3InO1xufVxuXG4vKiogVW5pZmllZCBjbGFzcyBpbXBsZW1lbnRpbmcgZGlmZmVyZW50IHN0cmluZyBtZWFzdXJlcy4gKi9cbmV4cG9ydCBjbGFzcyBNZWFzdXJlIHtcbiAgcHJvdGVjdGVkIG1ldGhvZDogS25vd25NZXRyaWNzO1xuICBwcm90ZWN0ZWQgZGF0YVR5cGU6IEF2YWlsYWJsZURhdGFUeXBlcztcblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiBpbnN0YW5jZSBvZiBNZWFzdXJlIHdpdGggLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gbWV0aG9kIE1ldGhvZCB0byBjYWxjdWxhdGUgZGlzdGFuY2UgYmV0d2VlbiBzdHJpbmdzLlxuICAgKiBAbWVtYmVyb2YgTWVhc3VyZXJcbiAgICovXG4gIGNvbnN0cnVjdG9yKG1ldGhvZDogS25vd25NZXRyaWNzKSB7XG4gICAgdGhpcy5tZXRob2QgPSBtZXRob2Q7XG4gICAgdGhpcy5kYXRhVHlwZSA9IE1ldHJpY1RvRGF0YVR5cGVbbWV0aG9kXSBhcyBBdmFpbGFibGVEYXRhVHlwZXM7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBjdXN0b20gc3RyaW5nIGRpc3RhbmNlIGZ1bmN0aW9uIHNwZWNpZmllZC5cbiAgICogQHJldHVybiB7RGlzdGFuY2VNZXRyaWN9IENhbGxiYWNrIG9mIHRoZSBtZWFzdXJlIGNob3Nlbi5cbiAgICogQG1lbWJlcm9mIE1lYXN1cmVyXG4gICAqL1xuICBwdWJsaWMgZ2V0TWVhc3VyZSgpOiBEaXN0YW5jZU1ldHJpYyB7XG4gICAgY29uc3QgZGljdDoge1trZXk6IHN0cmluZ106IHtba2V5Mjogc3RyaW5nXTogRGlzdGFuY2VNZXRyaWN9fSA9IEF2YWlsYWJsZU1ldHJpY3M7XG4gICAgcmV0dXJuIGRpY3RbdGhpcy5kYXRhVHlwZV1bdGhpcy5tZXRob2RdO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybnMgY3VzdG9tIHN0cmluZyBkaXN0YW5jZSBieSB0aGUgZ2l2ZW4gZGF0YSB0eXBlLlxuICAgKiBAcGFyYW0ge0F2YWlsYWJsZURhdGFUeXBlc30gZGF0YVR5cGUgTWV0cmljJ3MgZGF0YSB0eXBlXG4gICAqIEByZXR1cm4ge3N0cmluZ1tdfSBNZXRyaWMgbmFtZXMgd2hpY2ggZXhwZWN0cyB0aGUgZ2l2ZW4gZGF0YSB0eXBlXG4gICAqIEBtZW1iZXJvZiBNZWFzdXJlclxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBnZXRNZXRyaWNCeURhdGFUeXBlKGRhdGFUeXBlOiBBdmFpbGFibGVEYXRhVHlwZXMpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKEF2YWlsYWJsZU1ldHJpY3NbZGF0YVR5cGVdKTtcbiAgfVxuXG4gIC8qKiBSZXR1cm5zIG1ldHJpYyBuYW1lcyBhdmFpbGFibGUuXG4gICAqIEBtZW1iZXJvZiBNZWFzdXJlclxuICAqL1xuICBzdGF0aWMgZ2V0IGF2YWlsYWJsZU1lYXN1cmVzKCk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gT2JqZWN0LmtleXMoQXZhaWxhYmxlTWV0cmljcyk7XG4gIH1cbn1cbiJdfQ==","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { removeEmptyStringRows } from '@datagrok-libraries/utils/src/dataframe-utils';\nlet zoom = false;\nconst molColumnNames = ['1_seq', '2_seq'];\nconst nonNormalizedDistances = ['Levenshtein'];\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, activities, similarity, similarityMetric, methodName, semType, tags, seqSpaceFunc, simMatrixFunc, tooltipFunc, propertyPanelFunc, options) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const automaticSimilarityLimit = false;\n        const MIN_SIMILARITY = 80;\n        const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n        const dimensionalityReduceCol = encodedCol !== null && encodedCol !== void 0 ? encodedCol : seqCol;\n        const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n        //@ts-ignore\n        const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n        const { distance, coordinates } = yield seqSpaceFunc({\n            seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name),\n            methodName: methodName,\n            similarityMetric: similarityMetric,\n            embedAxesNames: axesNames,\n            options: options\n        });\n        for (const col of coordinates) {\n            const listValues = col.toList();\n            emptyValsIdxs.forEach((ind) => listValues.splice(ind, 0, null));\n            df.columns.add(DG.Column.fromList('double', col.name, listValues));\n        }\n        const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n        const dim = dimensionalityReduceCol.length;\n        const simArr = Array(dim - 1);\n        if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n            yield simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n        else\n            getSimilaritiesFromDistances(dim, distance, simArr);\n        const optSimilarityLimit = initialSimilarityLimit;\n        const simVals = [];\n        const saliVals = [];\n        const n1 = [];\n        const n2 = [];\n        for (let i = 0; i != dim - 1; ++i) {\n            for (let j = 0; j != dim - 1 - i; ++j) {\n                const sim = simArr[i] ? simArr[i].get(j) : 0;\n                if (sim >= optSimilarityLimit) {\n                    n1.push(i);\n                    n2.push(i + j + 1);\n                    simVals.push(sim);\n                    const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n                    if (sim != 1)\n                        saliVals.push(diff / (1 - sim));\n                    else\n                        saliVals.push(Infinity);\n                }\n            }\n        }\n        const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n        const saliMin = Math.min(...saliValsWithoutInfinity);\n        const saliMax = Math.max(...saliValsWithoutInfinity);\n        const saliOpacityCoef = 0.8 / (saliMax - saliMin);\n        const neighboursCount = new Array(dim).fill(0);\n        const similarityCount = new Array(dim).fill(0);\n        const saliCount = new Array(dim).fill(0);\n        for (let i = 0; i != n1.length; ++i) {\n            neighboursCount[n1[i]] += 1;\n            neighboursCount[n2[i]] += 1;\n            similarityCount[n1[i]] += simVals[i];\n            similarityCount[n2[i]] += simVals[i];\n            if (saliVals[i] != Infinity) {\n                if (activities.get(n1[i]) > activities.get(n2[i]))\n                    saliCount[n1[i]] += saliVals[i];\n                else\n                    saliCount[n2[i]] += saliVals[i];\n            }\n        }\n        const sali = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n        df.columns.add(sali);\n        const view = grok.shell.getTableView(df.name);\n        const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n            xColumnName: axesNames[0],\n            yColumnName: axesNames[1],\n            size: sali.name,\n            color: activities.name,\n            showXSelector: false,\n            showYSelector: false,\n            showSizeSelector: false,\n            showColorSelector: false,\n            markerMinSize: 5,\n            markerMaxSize: 25,\n            title: scatterTitle\n        }));\n        const canvas = sp.getInfo()['canvas'];\n        const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n        const cashedLinesData = {};\n        let acc;\n        linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n            const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n            const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n            sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n            sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n            sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n            linesDfGrid.invalidate();\n            if (line)\n                setTimeout(() => { updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc); }, 1000);\n        });\n        linesRes.linesDf.onSelectionChanged.subscribe((_) => {\n            if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n                const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n                line.selected = !line.selected;\n                if (!line.selected)\n                    df.selection.setAll(false);\n            }\n            linesRes.lines.forEach((l) => {\n                if (l.selected)\n                    l.mols.forEach((m) => df.selection.set(m, true));\n            });\n            linesDfGrid.invalidate();\n        });\n        const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n        linesDfGrid.onCellClick.subscribe(() => {\n            zoom = true;\n        });\n        const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n            grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n        });\n        listCliffsLink.style.position = 'absolute';\n        listCliffsLink.style.top = '10px';\n        listCliffsLink.style.right = '10px';\n        sp.root.append(listCliffsLink);\n        let timer;\n        canvas.addEventListener('mousemove', function (event) {\n            clearTimeout(timer);\n            timer = global.setTimeout(function () {\n                const line = checkCursorOnLine(event, canvas, linesRes.lines);\n                if (line && df.mouseOverRowIdx === -1) {\n                    ui.tooltip.show(tooltipFunc({ cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities }), event.clientX, event.clientY);\n                }\n            }, 500);\n        });\n        canvas.addEventListener('mousedown', function (event) {\n            const line = checkCursorOnLine(event, canvas, linesRes.lines);\n            if (line && df.mouseOverRowIdx === -1) {\n                if (event.ctrlKey) {\n                    line.selected = !line.selected;\n                    linesRes.linesDf.selection.set(line.id, line.selected);\n                    if (!line.selected)\n                        df.selection.setAll(false);\n                    linesRes.lines.forEach((l) => {\n                        if (l.selected)\n                            l.mols.forEach((m) => df.selection.set(m, true));\n                    });\n                }\n                else {\n                    if (linesRes.linesDf.currentRowIdx !== line.id) {\n                        linesRes.linesDf.currentRowIdx = line.id;\n                        df.currentRowIdx = line.mols[0];\n                        df.selection.set(0, !linesRes.lines[0].selected);\n                        df.selection.set(0, linesRes.lines[0].selected);\n                    }\n                }\n                const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n                linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n            }\n        });\n        sp.onEvent('d4-before-draw-scene')\n            .subscribe((_) => {\n            const lines = renderLines(sp, axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n            if (zoom) {\n                const currentLine = lines[linesRes.linesDf.currentRowIdx];\n                setTimeout(() => {\n                    const { zoomLeft, zoomRight, zoomTop, zoomBottom } = getZoomCoordinates(sp.viewport.width, sp.viewport.height, sp.dataFrame.get(axesNames[0], currentLine.mols[0]), sp.dataFrame.get(axesNames[1], currentLine.mols[0]), sp.dataFrame.get(axesNames[0], currentLine.mols[1]), sp.dataFrame.get(axesNames[1], currentLine.mols[1]));\n                    sp.zoom(zoomLeft, zoomTop, zoomRight, zoomBottom);\n                }, 300);\n                zoom = false;\n            }\n        });\n        sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n        acc = createPopertyPanel();\n        return sp;\n    });\n}\nfunction createPopertyPanel() {\n    const acc = ui.accordion();\n    const accIcon = ui.element('i');\n    accIcon.className = 'grok-icon svg-icon svg-view-layout';\n    acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n    acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n    grok.shell.o = acc.root;\n    return acc;\n}\nfunction updatePropertyPanel(df, acc, cashedData, line, seqCol, activities, sali, propPanelFunc) {\n    const panel = acc.getPane('Cliff Details');\n    ui.empty(panel.root);\n    const panelElement = propPanelFunc({ cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali });\n    panel.root.append(panelElement);\n    setTimeout(() => {\n        grok.shell.o = acc.root;\n    }, 500);\n}\nfunction getZoomCoordinates(W0, H0, x1, y1, x2, y2) {\n    const W1 = Math.abs(x1 - x2);\n    const H1 = Math.abs(y1 - y2);\n    const scaleW = W0 / W1;\n    const scaleH = H0 / H1;\n    const scale = Math.min(scaleW, scaleH);\n    const W2 = (W0 / scale) * 5;\n    const H2 = (H0 / scale) * 5;\n    const left = x1 < x2 ? x1 : x2;\n    const top = y1 > y2 ? y1 : y2;\n    const zoomLeft = (left + W1 / 2) - W2 / 2;\n    const zoomRight = zoomLeft + W2;\n    const zoomTop = (top - H1 / 2) + H2 / 2;\n    const zoomBottom = zoomTop - H2;\n    return { zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom };\n}\nfunction checkCursorOnLine(event, canvas, lines) {\n    const rect = canvas.getBoundingClientRect();\n    const x = event.clientX - rect.left;\n    const y = event.clientY - rect.top;\n    let closestLine = null;\n    let minDist = 0;\n    for (const line of lines) {\n        const dist = Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n            Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n        if ((!minDist && dist < 2) || dist < minDist) {\n            minDist = dist;\n            closestLine = line;\n        }\n    }\n    return closestLine;\n}\nfunction renderLines(sp, xAxis, yAxis, linesRes, saliVals, saliOpacityCoef, saliMin) {\n    const lines = linesRes.lines;\n    const canvas = sp.getInfo()['canvas'];\n    const ctx = canvas.getContext('2d');\n    const x = sp.dataFrame.columns.byName(xAxis);\n    const y = sp.dataFrame.columns.byName(yAxis);\n    for (let i = 0; i < lines.length; i++) {\n        const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n        const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n        lines[i].a = [pointFrom.x, pointFrom.y];\n        lines[i].b = [pointTo.x, pointTo.y];\n        const line = new Path2D();\n        line.moveTo(lines[i].a[0], lines[i].a[1]);\n        const color = lines[i].selected ? '255,255,0' : '0,128,0';\n        const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin) * saliOpacityCoef;\n        ctx.strokeStyle = `rgba(${color},${opacity})`;\n        ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n        line.lineTo(lines[i].b[0], lines[i].b[1]);\n        ctx.stroke(line);\n    }\n    return lines;\n}\nfunction createLines(n1, n2, seq, activities, saliVals, simVals, semType, tags) {\n    const lines = [];\n    for (let i = 0; i < n1.length; i++) {\n        const num1 = n1[i];\n        const num2 = n2[i];\n        lines.push({ id: i, mols: [num1, num2], selected: false, a: [], b: [] });\n    }\n    const linesDf = DG.DataFrame.create(lines.length);\n    molColumnNames.forEach((it, idx) => {\n        linesDf.columns.addNewString(it).init((i) => seq.get(lines[i].mols[idx]));\n        setTags(linesDf.col(it), tags);\n        linesDf.col(it).semType = semType;\n    });\n    linesDf.columns.addNewFloat('act_diff')\n        .init((i) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n    linesDf.columns.addNewInt('line_index').init((i) => i);\n    linesDf.columns.addNewFloat('sali').init((i) => saliVals[i]);\n    linesDf.columns.addNewFloat('sim').init((i) => simVals[i]);\n    return { lines, linesDf };\n}\nfunction setTags(col, tags) {\n    Object.keys(tags).forEach(tag => {\n        col.tags[tag] = tags[tag];\n    });\n}\nexport function getSimilaritiesMarix(dim, seqCol, dfSeq, simArr, simFunc) {\n    return __awaiter(this, void 0, void 0, function* () {\n        for (let i = 0; i != dim - 1; ++i) {\n            const mol = seqCol.get(i);\n            dfSeq.rows.removeAt(0, 1, false);\n            simArr[i] = (yield simFunc(dfSeq.col('seq'), mol));\n        }\n        return simArr;\n    });\n}\nexport function getSimilaritiesFromDistances(dim, distances, simArr) {\n    for (let i = 0; i < dim - 1; ++i) {\n        const similarityArr = new Array(dim - i - 1).fill(0);\n        for (let j = i + 1; j < dim; ++j) {\n            similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n        }\n        simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n    }\n    return simArr;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"activity-cliffs.js","sourceRoot":"","sources":["activity-cliffs.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,EAAE,yBAAyB,EAAE,MAAM,kDAAkD,CAAC;AAC7F,OAAO,EAAC,qBAAqB,EAAC,MAAM,+CAA+C,CAAA;AAqCnF,IAAI,IAAI,GAAG,KAAK,CAAC;AAEjB,MAAM,cAAc,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;AAE1C,MAAM,sBAAsB,GAAG,CAAC,aAAa,CAAC,CAAC;AAE/C,wEAAwE;AACxE,MAAM,UAAgB,iBAAiB,CACnC,EAAgB,EAChB,MAAiB,EACjB,UAA4B,EAC5B,SAAmB,EACnB,YAAoB,EACpB,UAAqB,EACrB,UAAkB,EAClB,gBAAwB,EACxB,UAAkB,EAClB,OAAe,EACf,IAA+B,EAC/B,YAA6E,EAC7E,aAA+H,EAC/H,WAA4D,EAC5D,iBAAkE,EAClE,OAAa;;QACf,MAAM,wBAAwB,GAAG,KAAK,CAAC;QACvC,MAAM,cAAc,GAAG,EAAE,CAAC;QAE1B,MAAM,sBAAsB,GAAG,wBAAwB,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,UAAU,GAAG,GAAG,CAAC;QAE5F,MAAM,uBAAuB,GAAG,UAAU,aAAV,UAAU,cAAV,UAAU,GAAI,MAAM,CAAC;QACrD,MAAM,kBAAkB,GAAG,EAAE,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,uBAAuB,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC;QACrF,YAAY;QACd,MAAM,aAAa,GAAG,qBAAqB,CAAC,kBAAkB,EAAE,uBAAuB,CAAC,CAAC;QAEzF,MAAM,EAAC,QAAQ,EAAE,WAAW,EAAC,GAAG,MAAM,YAAY,CAAC;YACjD,MAAM,EAAE,kBAAkB,CAAC,GAAG,CAAC,uBAAuB,CAAC,IAAI,CAAE;YAC7D,UAAU,EAAE,UAAU;YACtB,gBAAgB,EAAE,gBAAgB;YAClC,cAAc,EAAE,SAAS;YACzB,OAAO,EAAE,OAAO;SACjB,CAAC,CAAC;QAEH,KAAK,MAAM,GAAG,IAAI,WAAW,EAAE;YAC3B,MAAM,UAAU,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC;YAChC,aAAa,CAAC,OAAO,CAAC,CAAC,GAAW,EAAE,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;YACxE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ,EAAE,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC,CAAC;SACtE;QAED,MAAM,KAAK,GAAG,EAAE,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ,EAAE,KAAK,EAAE,uBAAuB,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,CAAC;QAChH,MAAM,GAAG,GAAG,uBAAuB,CAAC,MAAM,CAAC;QAC3C,MAAM,MAAM,GAAgB,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC;QAE3C,IAAI,CAAC,QAAQ,IAAI,aAAa,CAAC,MAAM,KAAK,CAAC,IAAI,sBAAsB,CAAC,QAAQ,CAAC,gBAAgB,CAAC;YAC9F,MAAM,aAAa,CAAC,GAAG,EAAE,uBAAuB,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;;YAExE,4BAA4B,CAAC,GAAG,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAEtD,MAAM,kBAAkB,GAAG,sBAAsB,CAAC;QAElD,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,MAAM,EAAE,GAAa,EAAE,CAAC;QACxB,MAAM,EAAE,GAAa,EAAE,CAAC;QAExB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,GAAG,GAAG,CAAC,EAAE,EAAE,CAAC,EAAE;YACjC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,GAAG,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,EAAE;gBACrC,MAAM,GAAG,GAAW,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAErD,IAAI,GAAG,IAAI,kBAAkB,EAAE;oBAC7B,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;oBACX,EAAE,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;oBACnB,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBAClB,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;oBACrE,IAAI,GAAG,IAAI,CAAC;wBACV,QAAQ,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC;;wBAEhC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBAC3B;aACF;SACF;QAED,MAAM,uBAAuB,GAAG,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,KAAK,QAAQ,CAAC,CAAC;QACvE,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,uBAAuB,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,uBAAuB,CAAC,CAAC;QACrD,MAAM,eAAe,GAAG,GAAG,GAAC,CAAC,OAAO,GAAG,OAAO,CAAC,CAAC;QAGhD,MAAM,eAAe,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC/C,MAAM,eAAe,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC/C,MAAM,SAAS,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAEzC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE;YACnC,eAAe,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAC5B,eAAe,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAC5B,eAAe,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC;YACrC,eAAe,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC;YACrC,IAAI,QAAQ,CAAC,CAAC,CAAC,IAAI,QAAQ,EAAE;gBAC3B,IAAI,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;oBAC/C,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,QAAQ,CAAC,CAAC,CAAC,CAAC;;oBAEhC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,QAAQ,CAAC,CAAC,CAAC,CAAC;aACnC;SACF;QAED,MAAM,IAAI,GAAc,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ,EAAE,QAAQ,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;QAEjI,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAErB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,EAAE;YAClD,WAAW,EAAE,SAAS,CAAC,CAAC,CAAC;YACzB,WAAW,EAAE,SAAS,CAAC,CAAC,CAAC;YACzB,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,KAAK,EAAE,UAAU,CAAC,IAAI;YACtB,aAAa,EAAE,KAAK;YACpB,aAAa,EAAE,KAAK;YACpB,gBAAgB,EAAE,KAAK;YACvB,iBAAiB,EAAE,KAAK;YACxB,aAAa,EAAE,CAAC;YAChB,aAAa,EAAE,EAAE;YACjB,KAAK,EAAE,YAAY;SACpB,CAAC,CAAyB,CAAC;QAE5B,MAAM,MAAM,GAAI,EAAE,CAAC,OAAO,EAAU,CAAC,QAAQ,CAAC,CAAC;QAC/C,MAAM,QAAQ,GAAG,WAAW,CAAC,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;QAC3F,MAAM,eAAe,GAAQ,EAAE,CAAC;QAChC,IAAI,GAAiB,CAAC;QAEtB,QAAQ,CAAC,OAAO,CAAC,oBAAoB,CAAC,SAAS,CAAC,GAAG,EAAE;YACnD,MAAM,aAAa,GAAG,QAAQ,CAAC,OAAO,CAAC,UAAU,IAAI,QAAQ,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAC1G,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,aAAa,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YAC3G,EAAE,CAAC,SAAS,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAClE,EAAE,CAAC,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YAC3D,EAAE,CAAC,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YAC1D,WAAW,CAAC,UAAU,EAAE,CAAC;YACzB,IAAI,IAAI;gBACN,UAAU,CAAC,GAAG,EAAE,GAAE,mBAAmB,CAAC,EAAE,EAAE,GAAG,EAAE,eAAe,EAAE,IAAI,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,CAAC,EAAE,iBAAiB,CAAC,CAAA,CAAA,CAAC,EAAE,IAAI,CAAC,CAAC;QAChK,CAAC,CAAC,CAAC;QAEH,QAAQ,CAAC,OAAO,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC,CAAM,EAAE,EAAE;YACvD,IAAI,QAAQ,CAAC,OAAO,CAAC,eAAe,KAAK,CAAC,CAAC,EAAE;gBAC3C,MAAM,IAAI,GAAG,QAAQ,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC;gBAC9D,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC;gBAC/B,IAAI,CAAC,IAAI,CAAC,QAAQ;oBAChB,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;aAC9B;YACD,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC3B,IAAI,CAAC,CAAC,QAAQ;oBACZ,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;YACrD,CAAC,CAAC,CAAC;YACH,WAAW,CAAC,UAAU,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;QAEH,MAAM,WAAW,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;QAEzE,WAAW,CAAC,WAAW,CAAC,SAAS,CAAC,GAAG,EAAE;YACrC,IAAI,GAAG,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;QAEH,MAAM,cAAc,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,QAAQ,CAAC,OAAO,CAAC,QAAQ,SAAS,EAAE,GAAG,EAAE;YAC3E,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,iBAAiB,EAAE,GAAG,CAAC,CAAC;QACtF,CAAC,CAAC,CAAC;QACH,cAAc,CAAC,KAAK,CAAC,QAAQ,GAAG,UAAU,CAAC;QAC3C,cAAc,CAAC,KAAK,CAAC,GAAG,GAAG,MAAM,CAAC;QAClC,cAAc,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;QACpC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;QAE/B,IAAI,KAAqB,CAAC;QAC1B,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,UAAU,KAAiB;YAC9D,YAAY,CAAC,KAAK,CAAC,CAAC;YACpB,KAAK,GAAG,MAAM,CAAC,UAAU,CAAC;gBACxB,MAAM,IAAI,GAAG,iBAAiB,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9D,IAAI,IAAI,IAAI,EAAE,CAAC,eAAe,KAAK,CAAC,CAAC,EAAE;oBACnC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,EAAC,UAAU,EAAE,eAAe,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,UAAU,EAAC,CAAC,EAAE,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;iBAC1J;YACH,CAAC,EAAE,GAAG,CAAC,CAAC;QACV,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,UAAS,KAAiB;YAC7D,MAAM,IAAI,GAAG,iBAAiB,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC;YAC9D,IAAI,IAAI,IAAI,EAAE,CAAC,eAAe,KAAK,CAAC,CAAC,EAAE;gBACrC,IAAI,KAAK,CAAC,OAAO,EAAE;oBACjB,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC;oBAC/B,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;oBAEvD,IAAI,CAAC,IAAI,CAAC,QAAQ;wBAChB,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;oBAE7B,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;wBAC3B,IAAI,CAAC,CAAC,QAAQ;4BACZ,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;oBACrD,CAAC,CAAC,CAAC;iBACJ;qBAAM;oBACL,IAAI,QAAQ,CAAC,OAAO,CAAC,aAAa,KAAK,IAAI,CAAC,EAAE,EAAE;wBAC9C,QAAQ,CAAC,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,EAAE,CAAC;wBACzC,EAAE,CAAC,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;wBAChC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;wBACjD,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;qBACjD;iBACF;gBACD,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,WAAW,CAAC,aAAa,EAAE,WAAW,CAAC,SAAS,CAAC,CAAC;gBAChG,WAAW,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC;aAC3D;QACH,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,OAAO,CAAC,sBAAsB,CAAC;aAC/B,SAAS,CAAC,CAAC,CAAM,EAAE,EAAE;YACpB,MAAM,KAAK,GAAG,WAAW,CAAC,EAAE,EAC1B,SAAS,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,eAAe,EAAE,OAAO,CAAC,CAAC;YAC5E,IAAI,IAAI,EAAE;gBACR,MAAM,WAAW,GAAG,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;gBAC1D,UAAU,CAAC,GAAE,EAAE;oBACb,MAAM,EAAC,QAAQ,EAAE,SAAS,EAAE,OAAO,EAAE,UAAU,EAAC,GAAG,kBAAkB,CACnE,EAAE,CAAC,QAAQ,CAAC,KAAK,EACjB,EAAE,CAAC,QAAQ,CAAC,MAAM,EAClB,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EACnD,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EACnD,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EACnD,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CACpD,CAAA;oBACD,EAAE,CAAC,IAAI,CAAC,QAAQ,EACd,OAAO,EACP,SAAS,EACT,UAAU,CAAC,CAAC;gBAChB,CAAC,EAAE,GAAG,CAAC,CAAC;gBACR,IAAI,GAAG,KAAK,CAAC;aACd;QACH,CAAC,CAAC,CAAC;QAEL,EAAE,CAAC,WAAW,CAAC,iBAAiB,EAAE,QAAQ,EAAE,kBAAkB,CAAC,CAAC;QAChE,GAAG,GAAG,kBAAkB,EAAE,CAAC;QAC3B,OAAO,EAAE,CAAC;IACZ,CAAC;CAAA;AAED,SAAS,kBAAkB;IACzB,MAAM,GAAG,GAAG,EAAE,CAAC,SAAS,EAAE,CAAC;IAC3B,MAAM,OAAO,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAChC,OAAO,CAAC,SAAS,GAAG,oCAAoC,CAAC;IACzD,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC;IAC9D,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC,OAAO,CAAC,6BAA6B,CAAC,EAAE,IAAI,CAAC,CAAC;IACpF,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,CAAC,IAAI,CAAC;IACxB,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAS,mBAAmB,CAC1B,EAAgB,EAChB,GAAiB,EACjB,UAAe,EACf,IAAW,EACX,MAAiB,EACjB,UAAqB,EACrB,IAAY,EACZ,aAA8D;IAC9D,MAAM,KAAK,GAAG,GAAG,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC;IAC3C,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACrB,MAAM,YAAY,GAAG,aAAa,CAAC,EAAC,UAAU,EAAE,UAAU,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,UAAU,EAAE,IAAI,EAAE,IAAI,EAAC,CAAC,CAAC;IACtI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IAChC,UAAU,CAAC,GAAG,EAAE;QACd,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,CAAC,IAAI,CAAC;IAC1B,CAAC,EAAE,GAAG,CAAC,CAAC;AACV,CAAC;AAED,SAAS,kBAAkB,CAAC,EAAU,EAAE,EAAU,EAAE,EAAU,EAAE,EAAU,EAAE,EAAU,EAAE,EAAU;IAChG,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,CAAC;IAC7B,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,CAAC;IAC7B,MAAM,MAAM,GAAG,EAAE,GAAG,EAAE,CAAC;IACvB,MAAM,MAAM,GAAG,EAAE,GAAG,EAAE,CAAC;IACvB,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACvC,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;IAC5B,MAAM,EAAE,GAAG,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;IAC5B,MAAM,IAAI,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IAC/B,MAAM,GAAG,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IAC9B,MAAM,QAAQ,GAAG,CAAC,IAAI,GAAG,EAAE,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IAC1C,MAAM,SAAS,GAAG,QAAQ,GAAG,EAAE,CAAC;IAChC,MAAM,OAAO,GAAG,CAAC,GAAG,GAAG,EAAE,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IACxC,MAAM,UAAU,GAAG,OAAO,GAAG,EAAE,CAAC;IAChC,OAAO,EAAC,QAAQ,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC;AAC9F,CAAC;AAED,SAAS,iBAAiB,CAAC,KAAU,EAAE,MAAW,EAAE,KAAc;IAChE,MAAM,IAAI,GAAG,MAAM,CAAC,qBAAqB,EAAE,CAAC;IAC5C,MAAM,CAAC,GAAG,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC;IACpC,MAAM,CAAC,GAAG,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC;IACnC,IAAI,WAAW,GAAG,IAAI,CAAC;IACvB,IAAI,OAAO,GAAG,CAAC,CAAC;IAChB,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;QACxB,MAAM,IAAI,GACR,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;YACjD,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACvG,IAAI,CAAC,CAAC,OAAO,IAAI,IAAI,GAAG,CAAC,CAAC,IAAI,IAAI,GAAG,OAAO,EAAE;YAC5C,OAAO,GAAG,IAAI,CAAC;YACf,WAAW,GAAG,IAAI,CAAC;SACpB;KACF;IACD,OAAO,WAAW,CAAC;AACrB,CAAC;AAED,SAAS,WAAW,CAAC,EAAwB,EAC3C,KAAa,EAAE,KAAa,EAAE,QAAwB,EAAE,QAAkB,EAAE,eAAuB,EAAE,OAAe;IACpH,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC;IAC7B,MAAM,MAAM,GAAI,EAAE,CAAC,OAAO,EAA8B,CAAC,QAAQ,CAAC,CAAC;IACnE,MAAM,GAAG,GAAG,MAAM,CAAC,UAAU,CAAC,IAAI,CAA6B,CAAC;IAChE,MAAM,CAAC,GAAG,EAAE,CAAC,SAAU,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAC9C,MAAM,CAAC,GAAG,EAAE,CAAC,SAAU,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAC9C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;QACrC,MAAM,SAAS,GAAG,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrF,MAAM,OAAO,GAAG,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnF,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC,CAAC,CAAC;QACxC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC;QACpC,MAAM,IAAI,GAAG,IAAI,MAAM,EAAE,CAAC;QAC1B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC;QAC1D,MAAM,OAAO,GAAG,QAAQ,CAAC,CAAC,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,GAAC,eAAe,CAAC;QAC7F,GAAG,CAAC,WAAW,GAAG,QAAQ,KAAK,IAAI,OAAO,GAAG,CAAC;QAC9C,GAAG,CAAC,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,QAAQ,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACvE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;KAClB;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,WAAW,CAClB,EAAY,EACZ,EAAY,EACZ,GAAc,EACd,UAAqB,EACrB,QAAkB,EAClB,OAAiB,EACjB,OAAe,EACf,IAA+B;IAC/B,MAAM,KAAK,GAAY,EAAE,CAAC;IAC1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;QAClC,MAAM,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QACnB,MAAM,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QACnB,KAAK,CAAC,IAAI,CAAC,EAAC,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC;KACxE;IACD,MAAM,OAAO,GAAG,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAClD,cAAc,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,EAAE;QACjC,OAAO,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QAClF,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAE,EAAE,IAAI,CAAC,CAAC;QAChC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAE,CAAC,OAAO,GAAG,OAAO,CAAC;IACrC,CAAC,CAAC,CAAA;IACF,OAAO,CAAC,OAAO,CAAC,WAAW,CAAC,UAAU,CAAC;SACpC,IAAI,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACtG,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;IAC/D,OAAO,CAAC,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;IACrE,OAAO,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;IACnE,OAAO,EAAC,KAAK,EAAE,OAAO,EAAC,CAAC;AAC1B,CAAC;AAED,SAAS,OAAO,CAAC,GAAc,EAAE,IAA+B;IAC9D,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;QAC9B,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,MAAM,UAAgB,oBAAoB,CACtC,GAAW,EACX,MAAiB,EACjB,KAAmB,EACnB,MAAmB,EACnB,OAAmE;;QAGrE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,GAAG,GAAG,CAAC,EAAE,EAAE,CAAC,EAAE;YACjC,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAC1B,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,KAAK,CAAC,CAAC;YACjC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAE,EAAE,GAAG,CAAC,CAAE,CAAC;SACtD;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;CAAA;AAGD,MAAM,UAAU,4BAA4B,CAAC,GAAW,EAAE,SAAiB,EAAE,MAAmB;IAE9F,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,GAAG,CAAC,EAAE,EAAE,CAAC,EAAE;QAChC,MAAM,aAAa,GAAG,IAAI,KAAK,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACrD,KAAK,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,EAAE,CAAC,EAAE;YAChC,aAAa,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,yBAAyB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SACvE;QACD,MAAM,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,gBAAgB,CAAC,YAAY,EAAE,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;KACxF;IACD,OAAO,MAAM,CAAC;AAChB,CAAC","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { Matrix } from '@datagrok-libraries/utils/src/type-declarations';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport {removeEmptyStringRows} from '@datagrok-libraries/utils/src/dataframe-utils'\n\nexport interface ILine {\n  id: number;\n  mols: number[];\n  selected: boolean;\n  a: number[]; // [x, y]\n  b: number[]; // [x, y]\n}\n\ninterface IRenderedLines {\n  lines: ILine[];\n  linesDf: DG.DataFrame;\n}\n\nexport interface ISequenceSpaceParams {\n  seqCol: DG.Column,\n  methodName: string,\n  similarityMetric: string,\n  embedAxesNames: string[],\n  options?: any\n}\n\nexport interface ISequenceSpaceResult {\n  distance: Matrix;\n  coordinates: DG.ColumnList;\n}\n\nexport interface ITooltipAndPanelParams {\n  cashedData: any,\n  line: ILine,\n  df: DG.DataFrame,\n  seqCol: DG.Column,\n  activityCol: DG.Column,\n  sali?: number\n}\n\nlet zoom = false;\n\nconst molColumnNames = ['1_seq', '2_seq'];\n\nconst nonNormalizedDistances = ['Levenshtein'];\n\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport async function getActivityCliffs(\n    df: DG.DataFrame, \n    seqCol: DG.Column,\n    encodedCol: DG.Column | null,\n    axesNames: string[],\n    scatterTitle: string,\n    activities: DG.Column, \n    similarity: number,\n    similarityMetric: string, \n    methodName: string, \n    semType: string,\n    tags: {[index: string]: string},\n    seqSpaceFunc: (params: ISequenceSpaceParams) => Promise<ISequenceSpaceResult>,\n    simMatrixFunc: (dim: number, seqCol: DG.Column, df: DG.DataFrame, colName: string, simArr: DG.Column[]) => Promise<DG.Column[]>,\n    tooltipFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    propertyPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    options?: any) : Promise<DG.Viewer> {\n  const automaticSimilarityLimit = false;\n  const MIN_SIMILARITY = 80;\n\n  const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n\n  const dimensionalityReduceCol = encodedCol ?? seqCol;\n  const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n    //@ts-ignore\n  const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n\n  const {distance, coordinates} = await seqSpaceFunc({\n    seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name)!,\n    methodName: methodName,\n    similarityMetric: similarityMetric,\n    embedAxesNames: axesNames,\n    options: options\n  });\n\n  for (const col of coordinates) {\n      const listValues = col.toList();\n      emptyValsIdxs.forEach((ind: number) => listValues.splice(ind, 0, null));\n      df.columns.add(DG.Column.fromList('double', col.name, listValues));\n  }\n\n  const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n  const dim = dimensionalityReduceCol.length;\n  const simArr: DG.Column[] = Array(dim - 1);\n\n  if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n    await simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n  else\n    getSimilaritiesFromDistances(dim, distance, simArr);\n\n  const optSimilarityLimit = initialSimilarityLimit;\n\n  const simVals: number[] = [];\n  const saliVals: number[] = [];\n  const n1: number[] = [];\n  const n2: number[] = [];\n\n  for (let i = 0; i != dim - 1; ++i) {\n    for (let j = 0; j != dim - 1 - i; ++j) {\n      const sim: number = simArr[i] ? simArr[i].get(j) : 0;\n\n      if (sim >= optSimilarityLimit) {\n        n1.push(i);\n        n2.push(i + j + 1);\n        simVals.push(sim);\n        const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n        if (sim != 1)\n          saliVals.push(diff / (1 - sim));\n        else\n          saliVals.push(Infinity);\n      }\n    }\n  }\n\n  const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n  const saliMin = Math.min(...saliValsWithoutInfinity);\n  const saliMax = Math.max(...saliValsWithoutInfinity);\n  const saliOpacityCoef = 0.8/(saliMax - saliMin);\n\n\n  const neighboursCount = new Array(dim).fill(0);\n  const similarityCount = new Array(dim).fill(0);\n  const saliCount = new Array(dim).fill(0);\n\n  for (let i = 0; i != n1.length; ++i) {\n    neighboursCount[n1[i]] += 1;\n    neighboursCount[n2[i]] += 1;\n    similarityCount[n1[i]] += simVals[i];\n    similarityCount[n2[i]] += simVals[i];\n    if (saliVals[i] != Infinity) {\n      if (activities.get(n1[i]) > activities.get(n2[i]))\n        saliCount[n1[i]] += saliVals[i];\n      else\n        saliCount[n2[i]] += saliVals[i];\n    }\n  }\n\n  const sali: DG.Column = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n\n  df.columns.add(sali);\n\n  const view = grok.shell.getTableView(df.name);\n  const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n    xColumnName: axesNames[0],\n    yColumnName: axesNames[1],\n    size: sali.name,\n    color: activities.name,\n    showXSelector: false,\n    showYSelector: false,\n    showSizeSelector: false,\n    showColorSelector: false,\n    markerMinSize: 5,\n    markerMaxSize: 25,\n    title: scatterTitle\n  })) as DG.ScatterPlotViewer;\n\n  const canvas = (sp.getInfo() as any)['canvas'];\n  const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n  const cashedLinesData: any = {};\n  let acc: DG.Accordion;\n\n  linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n    const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n    const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n    sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n    sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n    sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n    linesDfGrid.invalidate();\n    if (line)\n      setTimeout(() => {updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc)}, 1000);      \n  });\n\n  linesRes.linesDf.onSelectionChanged.subscribe((_: any) => {\n    if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n      const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n      line.selected = !line.selected;\n      if (!line.selected)\n        df.selection.setAll(false);\n    }\n    linesRes.lines.forEach((l) => {\n      if (l.selected)\n        l.mols.forEach((m) => df.selection.set(m, true));\n    });\n    linesDfGrid.invalidate();\n  });\n\n  const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n\n  linesDfGrid.onCellClick.subscribe(() => {\n    zoom = true;\n  });\n\n  const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n    grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n  });\n  listCliffsLink.style.position = 'absolute';\n  listCliffsLink.style.top = '10px';\n  listCliffsLink.style.right = '10px';\n  sp.root.append(listCliffsLink);\n\n  let timer: NodeJS.Timeout;\n  canvas.addEventListener('mousemove', function (event: MouseEvent) {\n    clearTimeout(timer);\n    timer = global.setTimeout(function () {\n      const line = checkCursorOnLine(event, canvas, linesRes.lines);\n      if (line && df.mouseOverRowIdx === -1) {\n          ui.tooltip.show(tooltipFunc({cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities}), event.clientX, event.clientY);\n      }\n    }, 500);\n  });\n\n  canvas.addEventListener('mousedown', function(event: MouseEvent) {\n    const line = checkCursorOnLine(event, canvas, linesRes.lines);\n    if (line && df.mouseOverRowIdx === -1) {\n      if (event.ctrlKey) {\n        line.selected = !line.selected;\n        linesRes.linesDf.selection.set(line.id, line.selected);\n\n        if (!line.selected)\n          df.selection.setAll(false);\n\n        linesRes.lines.forEach((l) => {\n          if (l.selected)\n            l.mols.forEach((m) => df.selection.set(m, true));\n        });\n      } else {\n        if (linesRes.linesDf.currentRowIdx !== line.id) {\n          linesRes.linesDf.currentRowIdx = line.id;\n          df.currentRowIdx = line.mols[0];\n          df.selection.set(0, !linesRes.lines[0].selected);\n          df.selection.set(0, linesRes.lines[0].selected);\n        }\n      }\n      const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n      linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n    }\n  });\n\n  sp.onEvent('d4-before-draw-scene')\n    .subscribe((_: any) => {\n      const lines = renderLines(sp,\n        axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n      if (zoom) {\n        const currentLine = lines[linesRes.linesDf.currentRowIdx];\n        setTimeout(()=> {\n          const {zoomLeft, zoomRight, zoomTop, zoomBottom} = getZoomCoordinates(\n            sp.viewport.width,\n            sp.viewport.height,\n            sp.dataFrame.get(axesNames[0], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[0], currentLine.mols[1]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[1])\n          )        \n          sp.zoom(zoomLeft,\n            zoomTop,\n            zoomRight,\n            zoomBottom);\n        }, 300);\n        zoom = false;\n      }\n    });\n\n  sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n  acc = createPopertyPanel();\n  return sp;\n}\n\nfunction createPopertyPanel(): DG.Accordion {\n  const acc = ui.accordion();\n  const accIcon = ui.element('i');\n  accIcon.className = 'grok-icon svg-icon svg-view-layout';\n  acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n  acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n  grok.shell.o = acc.root;\n  return acc;\n}\n\nfunction updatePropertyPanel(\n  df: DG.DataFrame,\n  acc: DG.Accordion,\n  cashedData: any,\n  line: ILine, \n  seqCol: DG.Column, \n  activities: DG.Column,\n  sali: number,\n  propPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement){\n  const panel = acc.getPane('Cliff Details');\n  ui.empty(panel.root);\n  const panelElement = propPanelFunc({cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali});\n  panel.root.append(panelElement);\n  setTimeout(() => {\n    grok.shell.o = acc.root;\n  }, 500);\n}\n\nfunction getZoomCoordinates(W0: number, H0: number, x1: number, y1: number, x2: number, y2: number) {\n  const W1 = Math.abs(x1 - x2);\n  const H1 = Math.abs(y1 - y2);\n  const scaleW = W0 / W1;\n  const scaleH = H0 / H1;\n  const scale = Math.min(scaleW, scaleH);\n  const W2 = (W0 / scale) * 5;\n  const H2 = (H0 / scale) * 5;\n  const left = x1 < x2 ? x1 : x2;\n  const top = y1 > y2 ? y1 : y2;\n  const zoomLeft = (left + W1 / 2) - W2 / 2;\n  const zoomRight = zoomLeft + W2;\n  const zoomTop = (top - H1 / 2) + H2 / 2;\n  const zoomBottom = zoomTop - H2;\n  return {zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom};\n}\n\nfunction checkCursorOnLine(event: any, canvas: any, lines: ILine[]): ILine | null {\n  const rect = canvas.getBoundingClientRect();\n  const x = event.clientX - rect.left;\n  const y = event.clientY - rect.top;\n  let closestLine = null;\n  let minDist = 0;\n  for (const line of lines) {\n    const dist =\n      Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n      Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n    if ((!minDist && dist < 2) || dist < minDist) {\n      minDist = dist;\n      closestLine = line;\n    }\n  }\n  return closestLine;\n}\n\nfunction renderLines(sp: DG.ScatterPlotViewer,\n  xAxis: string, yAxis: string, linesRes: IRenderedLines, saliVals: number[], saliOpacityCoef: number, saliMin: number): ILine [] {\n  const lines = linesRes.lines;\n  const canvas = (sp.getInfo() as {[index: string] : any})['canvas'];\n  const ctx = canvas.getContext('2d') as CanvasRenderingContext2D;\n  const x = sp.dataFrame!.columns.byName(xAxis);\n  const y = sp.dataFrame!.columns.byName(yAxis);\n  for (let i = 0; i < lines.length; i++) {\n    const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n    const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n    lines[i].a = [pointFrom.x, pointFrom.y];\n    lines[i].b = [pointTo.x, pointTo.y];\n    const line = new Path2D();\n    line.moveTo(lines[i].a[0], lines[i].a[1]);\n    const color = lines[i].selected ? '255,255,0' : '0,128,0';\n    const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin)*saliOpacityCoef;\n    ctx.strokeStyle = `rgba(${color},${opacity})`;\n    ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n    line.lineTo(lines[i].b[0], lines[i].b[1]);\n    ctx.stroke(line);\n  }\n  return lines;\n}\n\nfunction createLines(\n  n1: number[], \n  n2: number[], \n  seq: DG.Column, \n  activities: DG.Column, \n  saliVals: number[],\n  simVals: number[],\n  semType: string,\n  tags: {[index: string]: string}) : IRenderedLines {\n  const lines: ILine[] = [];\n  for (let i = 0; i < n1.length; i++) {\n    const num1 = n1[i];\n    const num2 = n2[i];\n    lines.push({id: i, mols: [num1, num2], selected: false, a: [], b: []});\n  }\n  const linesDf = DG.DataFrame.create(lines.length);\n  molColumnNames.forEach((it, idx) => {\n    linesDf.columns.addNewString(it).init((i: number) => seq.get(lines[i].mols[idx]));\n    setTags(linesDf.col(it)!, tags);\n    linesDf.col(it)!.semType = semType;\n  })\n  linesDf.columns.addNewFloat('act_diff')\n    .init((i: number) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n  linesDf.columns.addNewInt('line_index').init((i: number) => i);\n  linesDf.columns.addNewFloat('sali').init((i: number) => saliVals[i]);\n  linesDf.columns.addNewFloat('sim').init((i: number) => simVals[i]);\n  return {lines, linesDf};\n}\n\nfunction setTags(col: DG.Column, tags: {[index: string]: string}) {\n  Object.keys(tags).forEach(tag => {\n    col.tags[tag] = tags[tag];\n  })\n}\n\nexport async function getSimilaritiesMarix(\n    dim: number, \n    seqCol: DG.Column, \n    dfSeq: DG.DataFrame, \n    simArr: DG.Column[],\n    simFunc: (col: DG.Column, mol: string) => Promise<DG.Column | null>\n    )\n  : Promise<DG.Column[]> {\n  for (let i = 0; i != dim - 1; ++i) {\n    const mol = seqCol.get(i);\n    dfSeq.rows.removeAt(0, 1, false);\n    simArr[i] = (await simFunc(dfSeq.col('seq')!, mol))!;\n  }\n  return simArr;\n}\n\n\nexport function getSimilaritiesFromDistances(dim: number, distances: Matrix, simArr: DG.Column[])\n  : DG.Column[] {\n  for (let i = 0; i < dim - 1; ++i) {\n    const similarityArr = new Array(dim - i - 1).fill(0);\n    for (let j = i + 1; j < dim; ++j) {\n      similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n    }\n    simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n  }\n  return simArr;\n}\n\n"]}","export function createDimensinalityReducingWorker(dataMetric, method, options) {\n    return new Promise(function (resolve) {\n        const worker = new Worker(new URL('./dimensionality-reducer', import.meta.url));\n        worker.postMessage({\n            columnData: dataMetric.data,\n            method: method,\n            measure: dataMetric.metric,\n            options: options,\n        });\n        worker.onmessage = ({ data: { distance, embedding } }) => {\n            resolve({ distance: distance, embedding: embedding });\n        };\n    });\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGltZW5zaW9uYWxpdHktcmVkdWNpbmctd29ya2VyLWNyZWF0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJkaW1lbnNpb25hbGl0eS1yZWR1Y2luZy13b3JrZXItY3JlYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpQkEsTUFBTSxVQUFVLGlDQUFpQyxDQUFDLFVBQXNCLEVBQUUsTUFBYyxFQUNsRixPQUFhO0lBRWpCLE9BQU8sSUFBSSxPQUFPLENBQUMsVUFBUyxPQUFPO1FBQ2pDLE1BQU0sTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksR0FBRyxDQUFDLDBCQUEwQixFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUNoRixNQUFNLENBQUMsV0FBVyxDQUFDO1lBQ2pCLFVBQVUsRUFBRSxVQUFVLENBQUMsSUFBSTtZQUMzQixNQUFNLEVBQUUsTUFBTTtZQUNkLE9BQU8sRUFBRSxVQUFVLENBQUMsTUFBTTtZQUMxQixPQUFPLEVBQUUsT0FBTztTQUNqQixDQUFDLENBQUM7UUFDSCxNQUFNLENBQUMsU0FBUyxHQUFHLENBQUMsRUFBQyxJQUFJLEVBQUUsRUFBQyxRQUFRLEVBQUUsU0FBUyxFQUFDLEVBQUMsRUFBRSxFQUFFO1lBQ3BELE9BQU8sQ0FBQyxFQUFDLFFBQVEsRUFBRSxRQUFRLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBQyxDQUFDLENBQUM7UUFDckQsQ0FBQyxDQUFDO0lBQ0osQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWF0cml4IH0gZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvdHlwZS1kZWNsYXJhdGlvbnMnO1xuaW1wb3J0IHtWYWxpZFR5cGVzfSBmcm9tICcuLi90eXBlZC1tZXRyaWNzJztcblxuLyoqXG4gKiBBIHdvcmtlciB0byBwZXJmb3JtIGRpbWVuc2lvbmFsaXR5IHJlZHVjdGlvbi5cbiAqXG4gKiBAcGFyYW0ge1ZhbGlkVHlwZXN9IGRhdGFNZXRyaWMgVGhlIGRhdGEgdG8gcHJvY2Vzcy5cbiAqIEBwYXJhbSB7c3RyaW5nfSBtZXRob2QgQSBtZXRob2Qgb2YgZGltZW5zaW9uYWxpdHkgcmVkdWN0aW9uLlxuICogQHBhcmFtIG9wdGlvbnMgLSBrZXktdmFsdWUgcGFpcnNcbiAqIEBwYXJhbSByZXR1cm5EaXN0YW5jZU1hdHJpeFxuICogQHJldHVybiB7UHJvbWlzZTxJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQ+fSBSZXN1bHRpbmcgZW1iZWRkaW5nIGFuZCBkaXN0YW5jZSBtYXRyaXguXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVJlZHVjZURpbWVuc2lvbmFsaXR5UmVzdWx0IHtcbiAgZGlzdGFuY2U6IE1hdHJpeDtcbiAgZW1iZWRkaW5nOiBNYXRyaXg7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVEaW1lbnNpbmFsaXR5UmVkdWNpbmdXb3JrZXIoZGF0YU1ldHJpYzogVmFsaWRUeXBlcywgbWV0aG9kOiBzdHJpbmcsXG4gICAgICBvcHRpb25zPzogYW55KTogUHJvbWlzZTxJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQ+IHtcblxuICByZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24ocmVzb2x2ZSkge1xuICAgIGNvbnN0IHdvcmtlciA9IG5ldyBXb3JrZXIobmV3IFVSTCgnLi9kaW1lbnNpb25hbGl0eS1yZWR1Y2VyJywgaW1wb3J0Lm1ldGEudXJsKSk7XG4gICAgd29ya2VyLnBvc3RNZXNzYWdlKHtcbiAgICAgIGNvbHVtbkRhdGE6IGRhdGFNZXRyaWMuZGF0YSxcbiAgICAgIG1ldGhvZDogbWV0aG9kLFxuICAgICAgbWVhc3VyZTogZGF0YU1ldHJpYy5tZXRyaWMsXG4gICAgICBvcHRpb25zOiBvcHRpb25zLFxuICAgIH0pO1xuICAgIHdvcmtlci5vbm1lc3NhZ2UgPSAoe2RhdGE6IHtkaXN0YW5jZSwgZW1iZWRkaW5nfX0pID0+IHtcbiAgICAgcmVzb2x2ZSh7ZGlzdGFuY2U6IGRpc3RhbmNlLCBlbWJlZGRpbmc6IGVtYmVkZGluZ30pO1xuICAgIH07XG4gIH0pO1xufVxuIl19","import * as DG from 'datagrok-api/dg';\nimport * as sha256 from 'fast-sha256';\n/**\n * For columns of string type. Checks whether column contains empty values and removes corresponding rows in case user selects to remove.\n *\n */\nexport function removeEmptyStringRows(table, col) {\n    const cats = col.categories;\n    const emptyRawInd = cats.map((val, ind) => !val ? ind : null).filter((it) => it !== null);\n    const rawData = [...col.getRawData()];\n    const emptyRawsIndexes = [];\n    let removedRowsCounter = 0;\n    for (let i = 0; i < table.rowCount; i++) {\n        if (emptyRawInd.includes(rawData[i])) {\n            table.rows.removeAt(i - removedRowsCounter);\n            emptyRawsIndexes.push(i);\n            removedRowsCounter += 1;\n        }\n    }\n    return emptyRawsIndexes;\n}\nexport function hashDataFrame(table, names) {\n    names !== null && names !== void 0 ? names : (names = table.columns.names());\n    const hasher = new sha256.Hash();\n    const order = table.getSortedOrder(names);\n    const encoder = new TextEncoder();\n    for (const name of names) {\n        const column = table.columns.byName(name);\n        const dataArray = column.getRawData();\n        const isString = column.type == DG.TYPE.STRING;\n        const cats = column.categories;\n        for (let i = 0; i < dataArray.length; i++) {\n            if (isString) {\n                const data = cats[dataArray[order[i]]];\n                hasher.update(encoder.encode(data));\n            }\n            else {\n                const data = dataArray[order[i]];\n                hasher.update(Uint8Array.from([data]));\n            }\n        }\n    }\n    return hasher.digest();\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWZyYW1lLXV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZGF0YWZyYW1lLXV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDdEMsT0FBTyxLQUFLLE1BQU0sTUFBTSxhQUFhLENBQUM7QUFDdEM7OztHQUdHO0FBQ0gsTUFBTSxVQUFVLHFCQUFxQixDQUFDLEtBQW1CLEVBQUUsR0FBYztJQUN2RSxNQUFNLElBQUksR0FBRyxHQUFHLENBQUMsVUFBVSxDQUFDO0lBQzVCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRSxDQUFDLEVBQUUsS0FBSyxJQUFJLENBQUMsQ0FBQztJQUMxRixNQUFNLE9BQU8sR0FBRyxDQUFDLEdBQUcsR0FBRyxDQUFDLFVBQVUsRUFBRSxDQUFDLENBQUM7SUFDdEMsTUFBTSxnQkFBZ0IsR0FBRyxFQUFFLENBQUM7SUFDNUIsSUFBSSxrQkFBa0IsR0FBRyxDQUFDLENBQUM7SUFDM0IsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEtBQUssQ0FBQyxRQUFRLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDdkMsSUFBSSxXQUFXLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFO1lBQ3BDLEtBQUssQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDO1lBQzVDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN6QixrQkFBa0IsSUFBSSxDQUFDLENBQUM7U0FDekI7S0FDRjtJQUNELE9BQU8sZ0JBQWdCLENBQUM7QUFDMUIsQ0FBQztBQUVELE1BQU0sVUFBVSxhQUFhLENBQUMsS0FBbUIsRUFBRSxLQUFnQjtJQUNqRSxLQUFLLGFBQUwsS0FBSyxjQUFMLEtBQUssSUFBTCxLQUFLLEdBQUssS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsRUFBQztJQUNoQyxNQUFNLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNqQyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzFDLE1BQU0sT0FBTyxHQUFHLElBQUksV0FBVyxFQUFFLENBQUM7SUFDbEMsS0FBSyxNQUFNLElBQUksSUFBSSxLQUFLLEVBQUU7UUFDeEIsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDMUMsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ3RDLE1BQU0sUUFBUSxHQUFHLE1BQU0sQ0FBQyxJQUFJLElBQUksRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUM7UUFDL0MsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQztRQUMvQixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsU0FBUyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUN6QyxJQUFJLFFBQVEsRUFBRTtnQkFDWixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQ3ZDLE1BQU0sQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO2FBQ3JDO2lCQUFNO2dCQUNMLE1BQU0sSUFBSSxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztnQkFDakMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO2FBQ3hDO1NBQ0Y7S0FDRjtJQUNELE9BQU8sTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO0FBQ3pCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuaW1wb3J0ICogYXMgc2hhMjU2IGZyb20gJ2Zhc3Qtc2hhMjU2Jztcbi8qKlxuICogRm9yIGNvbHVtbnMgb2Ygc3RyaW5nIHR5cGUuIENoZWNrcyB3aGV0aGVyIGNvbHVtbiBjb250YWlucyBlbXB0eSB2YWx1ZXMgYW5kIHJlbW92ZXMgY29ycmVzcG9uZGluZyByb3dzIGluIGNhc2UgdXNlciBzZWxlY3RzIHRvIHJlbW92ZS5cbiAqXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiByZW1vdmVFbXB0eVN0cmluZ1Jvd3ModGFibGU6IERHLkRhdGFGcmFtZSwgY29sOiBERy5Db2x1bW4pOiBudW1iZXJbXSB7XG4gIGNvbnN0IGNhdHMgPSBjb2wuY2F0ZWdvcmllcztcbiAgY29uc3QgZW1wdHlSYXdJbmQgPSBjYXRzLm1hcCgodmFsLCBpbmQpID0+ICF2YWwgPyBpbmQgOiBudWxsKS5maWx0ZXIoKGl0KSA9PiBpdCAhPT0gbnVsbCk7XG4gIGNvbnN0IHJhd0RhdGEgPSBbLi4uY29sLmdldFJhd0RhdGEoKV07XG4gIGNvbnN0IGVtcHR5UmF3c0luZGV4ZXMgPSBbXTtcbiAgbGV0IHJlbW92ZWRSb3dzQ291bnRlciA9IDA7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgdGFibGUucm93Q291bnQ7IGkrKykge1xuICAgIGlmIChlbXB0eVJhd0luZC5pbmNsdWRlcyhyYXdEYXRhW2ldKSkge1xuICAgICAgdGFibGUucm93cy5yZW1vdmVBdChpIC0gcmVtb3ZlZFJvd3NDb3VudGVyKTtcbiAgICAgIGVtcHR5UmF3c0luZGV4ZXMucHVzaChpKTtcbiAgICAgIHJlbW92ZWRSb3dzQ291bnRlciArPSAxO1xuICAgIH1cbiAgfVxuICByZXR1cm4gZW1wdHlSYXdzSW5kZXhlcztcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGhhc2hEYXRhRnJhbWUodGFibGU6IERHLkRhdGFGcmFtZSwgbmFtZXM/OiBzdHJpbmdbXSk6IFVpbnQ4QXJyYXkge1xuICBuYW1lcyA/Pz0gdGFibGUuY29sdW1ucy5uYW1lcygpO1xuICBjb25zdCBoYXNoZXIgPSBuZXcgc2hhMjU2Lkhhc2goKTtcbiAgY29uc3Qgb3JkZXIgPSB0YWJsZS5nZXRTb3J0ZWRPcmRlcihuYW1lcyk7XG4gIGNvbnN0IGVuY29kZXIgPSBuZXcgVGV4dEVuY29kZXIoKTtcbiAgZm9yIChjb25zdCBuYW1lIG9mIG5hbWVzKSB7XG4gICAgY29uc3QgY29sdW1uID0gdGFibGUuY29sdW1ucy5ieU5hbWUobmFtZSk7XG4gICAgY29uc3QgZGF0YUFycmF5ID0gY29sdW1uLmdldFJhd0RhdGEoKTtcbiAgICBjb25zdCBpc1N0cmluZyA9IGNvbHVtbi50eXBlID09IERHLlRZUEUuU1RSSU5HO1xuICAgIGNvbnN0IGNhdHMgPSBjb2x1bW4uY2F0ZWdvcmllcztcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGRhdGFBcnJheS5sZW5ndGg7IGkrKykge1xuICAgICAgaWYgKGlzU3RyaW5nKSB7XG4gICAgICAgIGNvbnN0IGRhdGEgPSBjYXRzW2RhdGFBcnJheVtvcmRlcltpXV1dO1xuICAgICAgICBoYXNoZXIudXBkYXRlKGVuY29kZXIuZW5jb2RlKGRhdGEpKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IGRhdGEgPSBkYXRhQXJyYXlbb3JkZXJbaV1dO1xuICAgICAgICBoYXNoZXIudXBkYXRlKFVpbnQ4QXJyYXkuZnJvbShbZGF0YV0pKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcmV0dXJuIGhhc2hlci5kaWdlc3QoKTtcbn1cbiJdfQ==","/**\n * Generates single random float from 0 to range.\n *\n * @export\n * @param {number} range Max generating value.\n * @return {number} A random float generated.\n */\nexport function randomFloat(range) {\n    return Math.random() * range;\n}\n/**\n * Generates single random integer from 0 to range.\n *\n * @export\n * @param {number} range Max generating value.\n * @return {number} A random integer generated.\n */\nexport function randomInt(range) {\n    return Math.floor(randomFloat(range));\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFuZG9tLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicmFuZG9tLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxXQUFXLENBQUMsS0FBYTtJQUN2QyxPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxLQUFLLENBQUM7QUFDL0IsQ0FBQztBQUVEOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxTQUFTLENBQUMsS0FBYTtJQUNyQyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFDeEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVzIHNpbmdsZSByYW5kb20gZmxvYXQgZnJvbSAwIHRvIHJhbmdlLlxuICpcbiAqIEBleHBvcnRcbiAqIEBwYXJhbSB7bnVtYmVyfSByYW5nZSBNYXggZ2VuZXJhdGluZyB2YWx1ZS5cbiAqIEByZXR1cm4ge251bWJlcn0gQSByYW5kb20gZmxvYXQgZ2VuZXJhdGVkLlxuICovXG5leHBvcnQgZnVuY3Rpb24gcmFuZG9tRmxvYXQocmFuZ2U6IG51bWJlcik6IG51bWJlciB7XG4gIHJldHVybiBNYXRoLnJhbmRvbSgpICogcmFuZ2U7XG59XG5cbi8qKlxuICogR2VuZXJhdGVzIHNpbmdsZSByYW5kb20gaW50ZWdlciBmcm9tIDAgdG8gcmFuZ2UuXG4gKlxuICogQGV4cG9ydFxuICogQHBhcmFtIHtudW1iZXJ9IHJhbmdlIE1heCBnZW5lcmF0aW5nIHZhbHVlLlxuICogQHJldHVybiB7bnVtYmVyfSBBIHJhbmRvbSBpbnRlZ2VyIGdlbmVyYXRlZC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJhbmRvbUludChyYW5nZTogbnVtYmVyKTogbnVtYmVyIHtcbiAgcmV0dXJuIE1hdGguZmxvb3IocmFuZG9tRmxvYXQocmFuZ2UpKTtcbn1cbiJdfQ==","import { randomInt } from './random';\nexport const similarityMetric = {\n    'Tanimoto': tanimotoSimilarity,\n    'Dice': diceSimilarity,\n    'Asymmetric': asymmetricSimilarity,\n    'Braun-Blanquet': braunBlanquetSimilarity,\n    'Cosine': cosineSimilarity,\n    'Kulczynski': kulczynskiSimilarity,\n    'Mc-Connaughey': mcConnaugheySimilarity,\n    'Rogot-Goldberg': rogotGoldbergSimilarity,\n    'Russel': russelSimilarity,\n    'Sokal': sokalSimilarity,\n    'Hamming': hammingSimilarity,\n    'Euclidean': euclideanSimilarity,\n};\nexport const distanceMetrics = {\n    'Tanimoto': tanimotoDistance,\n    'Dice': diceDistance,\n    'Asymmetric': asymmetricDistance,\n    'Braun-Blanquet': braunBlanquetDistance,\n    'Cosine': cosineDistance,\n    'Kulczynski': kulczynskiDistance,\n    'Mc-Connaughey': mcConnaugheyDistance,\n    'Rogot-Goldberg': rogotGoldbergDistance,\n    'Russel': russelDistance,\n    'Sokal': sokalDistance,\n    'Hamming': hammingDistance,\n    'Euclidean': euclideanDistance,\n};\nexport const CHEM_SIMILARITY_METRICS = ['Tanimoto', 'Dice', 'Cosine', 'Hamming', 'Euclidean'];\nexport function tanimotoSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    if (total == 0)\n        return 1.0;\n    const common = x.andWithCountBits(y, true);\n    return common / (total - common);\n}\nexport function tanimotoDistance(x, y) {\n    return getDistanceFromSimilarity(tanimotoSimilarity(x, y));\n}\nexport function diceSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    if (total == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return 2 * common / total;\n}\nexport function diceDistance(x, y) {\n    return getDistanceFromSimilarity(diceSimilarity(x, y));\n}\nexport function cosineSimilarity(x, y) {\n    const total = x.trueCount() * y.trueCount();\n    if (total == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / Math.sqrt(total);\n}\nexport function cosineDistance(x, y) {\n    return getDistanceFromSimilarity(cosineSimilarity(x, y));\n}\nexport function euclideanSimilarity(x, y) {\n    return getSimilarityFromDistance(euclideanDistance(x, y));\n}\nexport function euclideanDistance(x, y) {\n    return Math.sqrt(x.trueCount() + y.trueCount() - 2 * x.andWithCountBits(y, true));\n}\nexport function hammingSimilarity(x, y) {\n    return getSimilarityFromDistance(hammingDistance(x, y));\n}\nexport function hammingDistance(x, y) {\n    return x.trueCount() + y.trueCount() - 2 * x.andWithCountBits(y, true);\n}\nexport function sokalSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const common = x.andWithCountBits(y, true);\n    return common / (2 * total - 3 * common);\n}\nexport function sokalDistance(x, y) {\n    return getDistanceFromSimilarity(sokalSimilarity(x, y));\n}\nexport function kulczynskiSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const totalProd = x.trueCount() * y.trueCount();\n    if (totalProd == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return (common * total) / (2 * totalProd);\n}\nexport function kulczynskiDistance(x, y) {\n    return getDistanceFromSimilarity(kulczynskiSimilarity(x, y));\n}\nexport function mcConnaugheySimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const totalProd = x.trueCount() * y.trueCount();\n    if (totalProd == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return (common * total - totalProd) / totalProd;\n}\nexport function mcConnaugheyDistance(x, y) {\n    return getDistanceFromSimilarity(mcConnaugheySimilarity(x, y));\n}\nexport function asymmetricSimilarity(x, y) {\n    const min = Math.min(x.trueCount(), y.trueCount());\n    if (min == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / min;\n}\nexport function asymmetricDistance(x, y) {\n    return getDistanceFromSimilarity(asymmetricSimilarity(x, y));\n}\nexport function braunBlanquetSimilarity(x, y) {\n    const max = Math.max(x.trueCount(), y.trueCount());\n    if (max == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / max;\n}\nexport function braunBlanquetDistance(x, y) {\n    return getDistanceFromSimilarity(braunBlanquetSimilarity(x, y));\n}\nexport function russelSimilarity(x, y) {\n    if (x.length == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / x.length;\n}\nexport function russelDistance(x, y) {\n    return getDistanceFromSimilarity(russelSimilarity(x, y));\n}\nexport function rogotGoldbergSimilarity(x, y) {\n    const common = x.andWithCountBits(y, true);\n    const total = x.countBits(true) + y.countBits(true);\n    const len = x.length;\n    const diff = len - total + common;\n    if ((common == len) || (diff == len))\n        return 1.0;\n    else\n        return common / total + diff / (2 * len - total);\n}\nexport function rogotGoldbergDistance(x, y) {\n    return getDistanceFromSimilarity(rogotGoldbergSimilarity(x, y));\n}\nexport function getSimilarityFromDistance(distance) {\n    return 1 / (1 + distance);\n}\nexport function getDistanceFromSimilarity(similarity) {\n    return 1 / similarity - 1;\n}\nexport function getDiverseSubset(length, n, dist) {\n    function maxBy(values, orderBy) {\n        let maxValue = null;\n        let maxOrderBy = null;\n        for (const element of values) {\n            const elementOrderBy = orderBy(element);\n            if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n                maxValue = element;\n                maxOrderBy = elementOrderBy;\n            }\n        }\n        return maxValue;\n    }\n    const subset = [randomInt(length - 1)];\n    const complement = new Set();\n    for (let i = 0; i < length; ++i) {\n        if (!subset.includes(i))\n            complement.add(i);\n    }\n    while (subset.length < n) {\n        const idx = maxBy(complement.values(), (i) => Math.min.apply(Math, subset.map(function (val, index) {\n            return dist(i, val);\n        })));\n        if (idx != null) {\n            subset.push(idx);\n            complement.delete(idx);\n        }\n    }\n    return subset;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"similarity-metrics.js","sourceRoot":"","sources":["similarity-metrics.ts"],"names":[],"mappings":"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","sourcesContent":["import BitArray from './bit-array';\nimport {randomInt} from './random';\n\nexport const similarityMetric: {[name: string]: (x: BitArray, y: BitArray) => number} = {\n  'Tanimoto': tanimotoSimilarity,\n  'Dice': diceSimilarity,\n  'Asymmetric': asymmetricSimilarity,\n  'Braun-Blanquet': braunBlanquetSimilarity,\n  'Cosine': cosineSimilarity,\n  'Kulczynski': kulczynskiSimilarity,\n  'Mc-Connaughey': mcConnaugheySimilarity,\n  'Rogot-Goldberg': rogotGoldbergSimilarity,\n  'Russel': russelSimilarity,\n  'Sokal': sokalSimilarity,\n  'Hamming': hammingSimilarity,\n  'Euclidean': euclideanSimilarity,\n};\n\nexport const distanceMetrics: {[name: string]: (x: BitArray, y: BitArray) => number} = {\n  'Tanimoto': tanimotoDistance,\n  'Dice': diceDistance,\n  'Asymmetric': asymmetricDistance,\n  'Braun-Blanquet': braunBlanquetDistance,\n  'Cosine': cosineDistance,\n  'Kulczynski': kulczynskiDistance,\n  'Mc-Connaughey': mcConnaugheyDistance,\n  'Rogot-Goldberg': rogotGoldbergDistance,\n  'Russel': russelDistance,\n  'Sokal': sokalDistance,\n  'Hamming': hammingDistance,\n  'Euclidean': euclideanDistance,\n};\n\nexport const CHEM_SIMILARITY_METRICS = ['Tanimoto', 'Dice', 'Cosine', 'Hamming', 'Euclidean'];\n\nexport function tanimotoSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  if (total == 0)\n    return 1.0;\n  const common = x.andWithCountBits(y, true);\n  return common / (total - common);\n}\n\nexport function tanimotoDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(tanimotoSimilarity(x, y));\n}\n\nexport function diceSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  if (total == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return 2 * common / total;\n}\n\nexport function diceDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(diceSimilarity(x, y));\n}\n\nexport function cosineSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() * y.trueCount();\n  if (total == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / Math.sqrt(total);\n}\n\nexport function cosineDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(cosineSimilarity(x, y));\n}\n\nexport function euclideanSimilarity(x: BitArray, y: BitArray): number {\n  return getSimilarityFromDistance(euclideanDistance(x, y));\n}\n\nexport function euclideanDistance(x: BitArray, y: BitArray): number {\n  return Math.sqrt(x.trueCount() + y.trueCount() - 2*x.andWithCountBits(y, true));\n}\n\nexport function hammingSimilarity(x: BitArray, y: BitArray): number {\n  return getSimilarityFromDistance(hammingDistance(x, y));\n}\n\nexport function hammingDistance(x: BitArray, y: BitArray): number {\n  return x.trueCount() + y.trueCount() - 2*x.andWithCountBits(y, true);\n}\n\nexport function sokalSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const common = x.andWithCountBits(y, true);\n  return common / (2 * total - 3 * common);\n}\n\nexport function sokalDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(sokalSimilarity(x, y));\n}\n\nexport function kulczynskiSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const totalProd = x.trueCount() * y.trueCount();\n  if (totalProd == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return (common * total) / (2 * totalProd);\n}\n\nexport function kulczynskiDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(kulczynskiSimilarity(x, y));\n}\n\nexport function mcConnaugheySimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const totalProd = x.trueCount() * y.trueCount();\n  if (totalProd == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return (common * total - totalProd) / totalProd;\n}\n\nexport function mcConnaugheyDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(mcConnaugheySimilarity(x, y));\n}\n\nexport function asymmetricSimilarity(x: BitArray, y: BitArray): number {\n  const min = Math.min(x.trueCount(), y.trueCount());\n  if (min == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / min;\n}\n\nexport function asymmetricDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(asymmetricSimilarity(x, y));\n}\n\nexport function braunBlanquetSimilarity(x: BitArray, y: BitArray): number {\n  const max = Math.max(x.trueCount(), y.trueCount());\n  if (max == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / max;\n}\n\nexport function braunBlanquetDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(braunBlanquetSimilarity(x, y));\n}\n\nexport function russelSimilarity(x: BitArray, y: BitArray): number {\n  if (x.length == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / x.length;\n}\n\nexport function russelDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(russelSimilarity(x, y));\n}\n\nexport function rogotGoldbergSimilarity(x: BitArray, y: BitArray): number {\n  const common = x.andWithCountBits(y, true);\n  const total = x.countBits(true) + y.countBits(true);\n  const len = x.length;\n  const diff = len - total + common;\n  if ((common == len) || (diff == len)) return 1.0;\n  else return common / total + diff / (2 * len - total);\n}\n\nexport function rogotGoldbergDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(rogotGoldbergSimilarity(x, y));\n}\n\nexport function getSimilarityFromDistance(distance: number) {\n  return 1 / ( 1 + distance);\n}\n\nexport function getDistanceFromSimilarity(similarity: number) {\n  return 1 / similarity - 1;\n}\n\nexport function getDiverseSubset(length: number, n: number, dist: (i1: number, i2: number) => number) {\n  function maxBy(values: IterableIterator<number>, orderBy: (i: number) => number) {\n    let maxValue = null;\n    let maxOrderBy = null;\n\n    for (const element of values) {\n      const elementOrderBy = orderBy(element);\n      if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n        maxValue = element;\n        maxOrderBy = elementOrderBy;\n      }\n    }\n    return maxValue;\n  }\n\n  const subset = [randomInt(length - 1)];\n  const complement = new Set();\n\n  for (let i = 0; i < length; ++i) {\n    if (!subset.includes(i))\n      complement.add(i);\n  }\n\n  while (subset.length < n) {\n    const idx = maxBy(\n      complement.values() as IterableIterator<number>,\n      (i) => Math.min.apply(Math, subset.map(function(val, index) {\n        return dist(i, val);\n      })));\n    if (idx != null) {\n      subset.push(idx);\n      complement.delete(idx);\n    }\n  }\n  return subset;\n}\n"]}","/**\n * Denotes a vector of floating poit values.\n *\n * @export\n * @class Vector\n * @extends {Float32Array}\n */\nexport class Vector extends Float32Array {\n}\n/**\n * Denotes a two-dimensional matrix.\n *\n * @export\n * @class Matrix\n * @extends {Array<Vector>}\n */\nexport class Matrix extends Array {\n}\n/**\n * Denotes cartesian coordinates.\n *\n * @export\n * @class Coordinates\n * @extends {Matrix}\n */\nexport class Coordinates extends Matrix {\n}\n/**\n * Denotes an array of arbitrary-typed vectors.\n *\n * @export\n * @class Vectors\n * @extends {Array<any>}\n */\nexport class Vectors extends Array {\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZS1kZWNsYXJhdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ0eXBlLWRlY2xhcmF0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sTUFBTyxTQUFRLFlBQVk7Q0FBRztBQUUzQzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sTUFBTyxTQUFRLEtBQWE7Q0FBRztBQUU1Qzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sV0FBWSxTQUFRLE1BQU07Q0FBRztBQUUxQzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sT0FBUSxTQUFRLEtBQVU7Q0FBRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogRGVub3RlcyBhIHZlY3RvciBvZiBmbG9hdGluZyBwb2l0IHZhbHVlcy5cbiAqXG4gKiBAZXhwb3J0XG4gKiBAY2xhc3MgVmVjdG9yXG4gKiBAZXh0ZW5kcyB7RmxvYXQzMkFycmF5fVxuICovXG5leHBvcnQgY2xhc3MgVmVjdG9yIGV4dGVuZHMgRmxvYXQzMkFycmF5IHt9XG5cbi8qKlxuICogRGVub3RlcyBhIHR3by1kaW1lbnNpb25hbCBtYXRyaXguXG4gKlxuICogQGV4cG9ydFxuICogQGNsYXNzIE1hdHJpeFxuICogQGV4dGVuZHMge0FycmF5PFZlY3Rvcj59XG4gKi9cbmV4cG9ydCBjbGFzcyBNYXRyaXggZXh0ZW5kcyBBcnJheTxWZWN0b3I+IHt9XG5cbi8qKlxuICogRGVub3RlcyBjYXJ0ZXNpYW4gY29vcmRpbmF0ZXMuXG4gKlxuICogQGV4cG9ydFxuICogQGNsYXNzIENvb3JkaW5hdGVzXG4gKiBAZXh0ZW5kcyB7TWF0cml4fVxuICovXG5leHBvcnQgY2xhc3MgQ29vcmRpbmF0ZXMgZXh0ZW5kcyBNYXRyaXgge31cblxuLyoqXG4gKiBEZW5vdGVzIGFuIGFycmF5IG9mIGFyYml0cmFyeS10eXBlZCB2ZWN0b3JzLlxuICpcbiAqIEBleHBvcnRcbiAqIEBjbGFzcyBWZWN0b3JzXG4gKiBAZXh0ZW5kcyB7QXJyYXk8YW55Pn1cbiAqL1xuZXhwb3J0IGNsYXNzIFZlY3RvcnMgZXh0ZW5kcyBBcnJheTxhbnk+IHt9XG5cbi8qKlxuICogRGVub3RlcyBhIGRpY3Rpb25hcnkgY29udGFpbmluZyBmdW5jdGlvbiBvcHRpb25zLlxuICpcbiAqIEBleHBvcnRcbiAqIEB0eXBlIE9wdGlvbnNcbiAqL1xuZXhwb3J0IHR5cGUgT3B0aW9ucyA9IHtbbmFtZTogc3RyaW5nXTogYW55fTtcblxuLyoqXG4gKiBEZW5vdGVzIGN1c3RvbSBkaXN0YW5jZSBtZXRyaWMgYmV0d2VlbiB0aGUgdHdvIGdpdmVuIHZlY3RvcnMuXG4gKlxuICogQGV4cG9ydFxuICogQHR5cGUgRGlzdGFuY2VNZXRyaWNcbiAqIEBwYXJhbSB7YW55fSB2MSBUaGUgZmlyc3QgdmVjdG9yLlxuICogQHBhcmFtIHthbnl9IHYyIFRoZSBzZWNvbmQgdmVjdG9yLlxuICogQHJldHVybiB7bnVtYmVyfSBEaXN0YW5jZSBiZXR3ZWVuIHRoZXNlIHR3byB2ZWN0b3JzLlxuICovXG5leHBvcnQgdHlwZSBEaXN0YW5jZU1ldHJpYyA9ICh2MTogYW55LCB2MjogYW55KSA9PiAobnVtYmVyKTtcblxuLyoqXG4gKiBEZW5vdGVzIGEgc2ltcGxlIHN0cmluZyB0byBzdHJpbmcgZGljdGlvbmFyeS5cbiAqXG4gKiBAZXhwb3J0XG4gKiBAdHlwZSBTdHJpbmdEaWN0aW9uYXJ5XG4gKi9cbmV4cG9ydCB0eXBlIFN0cmluZ0RpY3Rpb25hcnkgPSB7W2tleTogc3RyaW5nXTogc3RyaW5nfTtcbiJdfQ==","import { Vector } from './type-declarations';\nimport { randomFloat, randomInt } from './random';\n/**\n * Asserts a condition by throwing an Error.\n *\n * @export\n * @param {boolean} [condition=false] Condition to assert.\n * @param {string} [message='Assertion error.'] Message to output.\n * @throws {Error}\n */\nexport function assert(condition = false, message = 'Assertion error.') {\n    if (!condition)\n        throw new Error(message);\n}\n/**\n * Creates new two-dimensional array and fills it with the value given.\n *\n * @param {number} dimension1 The first dimension of the coordinates (number of rows).\n * @param {number} dimension2 The second dimension of the coordinates (number of columns).\n * @param {number} [fill=0] A value to fill the coordinates with.\n * @return {Coordinates} A two-dimensional filled with the value given.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nfunction initCoordinates(dimension1, dimension2, fill = 0) {\n    return new Array(dimension1).fill(fill).map(() => (new Vector(dimension2).fill(fill)));\n}\n/**\n * Transpose matrix.\n *\n * @export\n * @param {Matrix} matrix The matrix to be transposed.\n * @return {Matrix} Transposed matrix.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nexport function transposeMatrix(matrix) {\n    return new Array(matrix[0].length).fill(0)\n        .map((_, i) => (new Vector(matrix.length).fill(0).map((_, j) => (matrix[j][i]))));\n}\n/**\n * Adds two vectors with the second one to be multiplied by the given ratio.\n *\n * @export\n * @param {Vector} p The first vector to add.\n * @param {Vector} q The second vector to add.\n * @param {number} [multiplier=1] A multiplier to be used before the second vector is added.\n * @return {Vector} New vector contained the result of operation p+multiplier*q.\n */\nexport function vectorAdd(p, q, multiplier = 1) {\n    const nItems = p.length;\n    assert(nItems == q.length, 'Vector lengths do not match.');\n    const total = new Vector(nItems);\n    for (let i = 0; i < p.length; ++i)\n        total[i] = p[i] + multiplier * q[i];\n    return total;\n}\n/**\n * Sums the vector's items.\n *\n * @param {Vector} v The vector to be summed.\n * @return {number} The vector's items sum.\n */\nfunction itemsSum(v) {\n    let total = 0;\n    for (let i = 0; i < v.length; ++i)\n        total += v[i];\n    return total;\n}\n/**\n * Suqares the vector's items.\n *\n * @param {Vector} v The vector to square.\n * @return {Vector} A new vector containing the original's items squared.\n */\nfunction vectorSquare(v) {\n    const nItems = v.length;\n    const total = new Vector(nItems);\n    for (let i = 0; i < v.length; ++i)\n        total[i] = v[i] * v[i];\n    return total;\n}\nexport function vectorLength(v) {\n    let sqrSum = 0;\n    for (let i = 0; i < v.length; i++)\n        sqrSum += v[i] * v[i];\n    return Math.sqrt(sqrSum);\n}\nexport function vectorDotProduct(v1, v2) {\n    if (v1.length != v2.length)\n        throw Error('The dimensionality of the vectors must match');\n    let prod = 0;\n    for (let i = 0; i < v1.length; i++)\n        prod += v1[i] * v2[i];\n    return prod;\n}\n/**\n * Creates a matrix filled with random floating point values.\n *\n * @export\n * @param {number} dimension1 The first dimension of the matrix.\n * @param {number} dimension2 The second dimension of the matrix.\n * @param {number} [scale=1.] Max value given by random generator.\n * @return {Matrix} A new matrix filled with random floating point  values.\n */\nexport function fillRandomMatrix(dimension1, dimension2, scale = 1.) {\n    const matrix = initCoordinates(dimension1, dimension2);\n    for (let i = 0; i < dimension1; ++i) {\n        for (let j = 0; j < dimension2; ++j)\n            matrix[i][j] = randomFloat(scale);\n    }\n    return matrix;\n}\n/**\n * Calculates Euclidean distance between two vectors.\n *\n * @export\n * @param {Vector} p The first vector.\n * @param {Vector} q The second vector.\n * @return {number} Euclidean distance between the given vectors.\n */\nexport function calculateEuclideanDistance(p, q) {\n    const diff = vectorAdd(p, q, -1);\n    const sqdiff = vectorSquare(diff);\n    const sqdiffSumm = itemsSum(sqdiff);\n    return Math.sqrt(sqdiffSumm);\n}\n/**\n * Creates a distance matrix using a custom distance function.\n *\n * @export\n * @param {Vectors} data Input vectors to calculate distances.\n * @param {DistanceMetric} distance Custom distance function.\n * @return {Matrix} Calculated custom distance matrix.\n */\nexport function calcDistanceMatrix(data, distance) {\n    const nItems = data.length;\n    const matrix = initCoordinates(nItems, nItems, 0);\n    for (let i = 0; i < nItems; ++i) {\n        for (let j = i + 1; j < nItems; ++j) {\n            const d = (data[i] == null) || (data[j] == null) ? 0 : distance(data[i], data[j]);\n            matrix[i][j] = matrix[j][i] = d;\n        }\n    }\n    return matrix;\n}\n/** Generates array from a range [begin; end] or [begin; end) if endExclusive. **/\nexport function genRange(begin, end, endExclusive = false) {\n    const nItems = end - begin + (endExclusive ? 0 : 1);\n    const series = new Int32Array(nItems);\n    for (let i = 0; i < nItems; ++i)\n        series[i] = begin + i;\n    return series;\n}\n/**\n * Returns order of values as if they are sorted.\n *\n * @export\n * @param {any[]} values Input array.\n * @param {boolean} [reverse=false] Whether to return reversed order.\n * @return {number[]} The order computed.\n */\nexport function argSort(values, reverse = false) {\n    const sortfn = reverse ? (a, b) => (b[0] - a[0]) : (a, b) => (a[0] - b[0]);\n    const decor = (v, i) => [v, i]; // set index to value\n    const undecor = (a) => a[1]; // leave only index\n    const _argsort = (arr) => arr.map(decor).sort(sortfn).map(undecor);\n    return _argsort(values);\n}\n/**\n * Returns the indexes of the most diverse objects according to the dist function\n * @param {number} length total number of objects\n * @param {number} n number of diverse elements to find\n * @param {(i1: number, i2: number) => number} dist a function which calculates distance between\n *                                                  two objects using their indexes\n * @returns {number[]} The indexes of the most diverse objects\n */\nexport function getDiverseSubset(length, n, dist) {\n    function maxBy(values, orderBy) {\n        let maxValue = null;\n        let maxOrderBy = null;\n        for (const element of values) {\n            const elementOrderBy = orderBy(element);\n            if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n                maxValue = element;\n                maxOrderBy = elementOrderBy;\n            }\n        }\n        return maxValue;\n    }\n    const subset = [randomInt(length - 1)];\n    const complement = new Set();\n    for (let i = 0; i < length; ++i) {\n        if (!subset.includes(i))\n            complement.add(i);\n    }\n    while (subset.length < n) {\n        const idx = maxBy(complement.values(), (i) => Math.min.apply(Math, subset.map(function (val, index) {\n            return dist(i, val);\n        })));\n        if (idx) {\n            subset.push(idx);\n            complement.delete(idx);\n        }\n    }\n    return subset;\n}\n/**\n * Returns normalized vector\n * @param {Vector} data numerical array\n */\nexport function normalize(data) {\n    let mean = 0;\n    let std = 0;\n    for (let i = 0; i < data.length; ++i)\n        mean += data[i];\n    mean /= data.length;\n    for (let i = 0; i < data.length; ++i)\n        std += (data[i] - mean) * (data[i] - mean);\n    std = Math.sqrt(std / data.length);\n    for (let i = 0; i < data.length; ++i)\n        data[i] = (data[i] - mean) / std;\n    return data;\n}\n/**\n * Finds set difference between two lists.\n * @param {any[]} a The first list.\n * @param {any[]} b The second list.\n * @return {any[]}\n */\nexport function setDifference(a, b) {\n    const bSet = new Set(b);\n    return Array.from(new Set(a.filter((x) => !bSet.has(x))).values());\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vector-operations.js","sourceRoot":"","sources":["vector-operations.ts"],"names":[],"mappings":"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","sourcesContent":["import {Matrix, Vector, Coordinates, Vectors, DistanceMetric} from './type-declarations';\nimport {randomFloat, randomInt} from './random';\n\n/**\n * Asserts a condition by throwing an Error.\n *\n * @export\n * @param {boolean} [condition=false] Condition to assert.\n * @param {string} [message='Assertion error.'] Message to output.\n * @throws {Error}\n */\nexport function assert(condition: boolean = false, message: string = 'Assertion error.') {\n  if (!condition)\n    throw new Error(message);\n}\n\n/**\n * Creates new two-dimensional array and fills it with the value given.\n *\n * @param {number} dimension1 The first dimension of the coordinates (number of rows).\n * @param {number} dimension2 The second dimension of the coordinates (number of columns).\n * @param {number} [fill=0] A value to fill the coordinates with.\n * @return {Coordinates} A two-dimensional filled with the value given.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nfunction initCoordinates(dimension1: number, dimension2: number, fill: number = 0): Coordinates {\n  return new Array(dimension1).fill(fill).map(() => (new Vector(dimension2).fill(fill)));\n}\n\n/**\n * Transpose matrix.\n *\n * @export\n * @param {Matrix} matrix The matrix to be transposed.\n * @return {Matrix} Transposed matrix.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nexport function transposeMatrix(matrix: Matrix): Matrix {\n  return new Array(matrix[0].length).fill(0)\n    .map((_, i) => (new Vector(matrix.length).fill(0).map((_, j) => (matrix[j][i]))));\n}\n\n/**\n * Adds two vectors with the second one to be multiplied by the given ratio.\n *\n * @export\n * @param {Vector} p The first vector to add.\n * @param {Vector} q The second vector to add.\n * @param {number} [multiplier=1] A multiplier to be used before the second vector is added.\n * @return {Vector} New vector contained the result of operation p+multiplier*q.\n */\nexport function vectorAdd(p: Vector, q: Vector, multiplier: number = 1): Vector {\n  const nItems = p.length;\n\n  assert(nItems == q.length, 'Vector lengths do not match.');\n\n  const total = new Vector(nItems);\n\n  for (let i = 0; i < p.length; ++i)\n    total[i] = p[i] + multiplier * q[i];\n\n  return total;\n}\n\n/**\n * Sums the vector's items.\n *\n * @param {Vector} v The vector to be summed.\n * @return {number} The vector's items sum.\n */\nfunction itemsSum(v: Vector): number {\n  let total = 0;\n\n  for (let i = 0; i < v.length; ++i)\n    total += v[i];\n\n  return total;\n}\n\n/**\n * Suqares the vector's items.\n *\n * @param {Vector} v The vector to square.\n * @return {Vector} A new vector containing the original's items squared.\n */\nfunction vectorSquare(v: Vector): Vector {\n  const nItems = v.length;\n  const total = new Vector(nItems);\n\n  for (let i = 0; i < v.length; ++i)\n    total[i] = v[i] * v[i];\n\n  return total;\n}\n\nexport function vectorLength(v: Vector): number {\n  let sqrSum: number = 0;\n  for (let i: number = 0; i < v.length; i++)\n    sqrSum += v[i] * v[i];\n  return Math.sqrt(sqrSum);\n}\n\nexport function vectorDotProduct(v1: Vector, v2: Vector): number {\n  if (v1.length != v2.length)\n    throw Error('The dimensionality of the vectors must match');\n  let prod: number = 0;\n  for (let i: number = 0; i < v1.length; i++)\n    prod += v1[i] * v2[i];\n  return prod;\n}\n\n/**\n * Creates a matrix filled with random floating point values.\n *\n * @export\n * @param {number} dimension1 The first dimension of the matrix.\n * @param {number} dimension2 The second dimension of the matrix.\n * @param {number} [scale=1.] Max value given by random generator.\n * @return {Matrix} A new matrix filled with random floating point  values.\n */\nexport function fillRandomMatrix(dimension1: number, dimension2: number, scale: number = 1.): Matrix {\n  const matrix = initCoordinates(dimension1, dimension2);\n\n  for (let i = 0; i < dimension1; ++i) {\n    for (let j = 0; j < dimension2; ++j)\n      matrix[i][j] = randomFloat(scale);\n  }\n  return matrix;\n}\n\n/**\n * Calculates Euclidean distance between two vectors.\n *\n * @export\n * @param {Vector} p The first vector.\n * @param {Vector} q The second vector.\n * @return {number} Euclidean distance between the given vectors.\n */\nexport function calculateEuclideanDistance(p: Vector, q: Vector): number {\n  const diff = vectorAdd(p, q, -1);\n  const sqdiff = vectorSquare(diff);\n  const sqdiffSumm = itemsSum(sqdiff);\n  return Math.sqrt(sqdiffSumm);\n}\n\n/**\n * Creates a distance matrix using a custom distance function.\n *\n * @export\n * @param {Vectors} data Input vectors to calculate distances.\n * @param {DistanceMetric} distance Custom distance function.\n * @return {Matrix} Calculated custom distance matrix.\n */\nexport function calcDistanceMatrix(data: Vectors, distance: DistanceMetric): Matrix {\n  const nItems = data.length;\n  const matrix = initCoordinates(nItems, nItems, 0);\n\n  for (let i = 0; i < nItems; ++i) {\n    for (let j = i + 1; j < nItems; ++j) {\n      const d: number = (data[i] == null) || (data[j] == null) ? 0 : distance(data[i], data[j]);\n      matrix[i][j] = matrix[j][i] = d;\n    }\n  }\n  return matrix;\n}\n\n/** Generates array from a range [begin; end] or [begin; end) if endExclusive. **/\nexport function genRange(begin: number, end: number, endExclusive = false): Int32Array {\n  const nItems = end - begin + (endExclusive ? 0 : 1);\n  const series = new Int32Array(nItems);\n\n  for (let i = 0; i < nItems; ++i)\n    series[i] = begin + i;\n\n  return series;\n}\n\n/**\n * Returns order of values as if they are sorted.\n *\n * @export\n * @param {any[]} values Input array.\n * @param {boolean} [reverse=false] Whether to return reversed order.\n * @return {number[]} The order computed.\n */\nexport function argSort(values: any[], reverse = false): number[] {\n  const sortfn = reverse ? (a: any[], b: any[]) => (b[0] - a[0]) : (a: any[], b: any[]) => (a[0] - b[0]);\n  const decor = (v: any, i: number) => [v, i]; // set index to value\n  const undecor = (a: any[]) => a[1]; // leave only index\n  const _argsort = (arr: any[]) => arr.map(decor).sort(sortfn).map(undecor);\n  return _argsort(values);\n}\n\n/**\n * Returns the indexes of the most diverse objects according to the dist function\n * @param {number} length total number of objects\n * @param {number} n number of diverse elements to find\n * @param {(i1: number, i2: number) => number} dist a function which calculates distance between\n *                                                  two objects using their indexes\n * @returns {number[]} The indexes of the most diverse objects\n */\nexport function getDiverseSubset(length: number, n: number, dist: (i1: number, i2: number) => number): number[] {\n  function maxBy(values: IterableIterator<number>, orderBy: (i: number) => number) {\n    let maxValue = null;\n    let maxOrderBy = null;\n\n    for (const element of values) {\n      const elementOrderBy = orderBy(element);\n      if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n        maxValue = element;\n        maxOrderBy = elementOrderBy;\n      }\n    }\n    return maxValue;\n  }\n\n  const subset = [randomInt(length - 1)];\n  const complement = new Set();\n\n  for (let i = 0; i < length; ++i) {\n    if (!subset.includes(i))\n      complement.add(i);\n  }\n\n  while (subset.length < n) {\n    const idx = maxBy(\n      complement.values() as IterableIterator<number>,\n      (i) => Math.min.apply(Math, subset.map(function(val, index) {\n        return dist(i, val);\n      })));\n    if (idx) {\n      subset.push(idx);\n      complement.delete(idx);\n    }\n  }\n  return subset;\n}\n\n/**\n * Returns normalized vector\n * @param {Vector} data numerical array\n */\nexport function normalize(data: Vector): Vector {\n  let mean = 0;\n  let std = 0;\n\n  for (let i = 0; i < data.length; ++i)\n    mean += data[i];\n\n  mean /= data.length;\n\n  for (let i = 0; i < data.length; ++i)\n    std += (data[i] - mean) * (data[i] - mean);\n\n  std = Math.sqrt(std / data.length);\n\n  for (let i = 0; i < data.length; ++i)\n    data[i] = (data[i] - mean) / std;\n\n  return data;\n}\n\n/**\n * Finds set difference between two lists.\n * @param {any[]} a The first list.\n * @param {any[]} b The second list.\n * @return {any[]}\n */\nexport function setDifference(a: any[], b: any[]): any[] {\n  const bSet = new Set(b);\n  return Array.from(new Set(a.filter((x) => !bSet.has(x))).values());\n}\n"]}","(function (root, factory) {\n    // Hack to make all exports of this module sha256 function object properties.\n    var exports = {};\n    factory(exports);\n    var sha256 = exports[\"default\"];\n    for (var k in exports) {\n        sha256[k] = exports[k];\n    }\n        \n    if (typeof module === 'object' && typeof module.exports === 'object') {\n        module.exports = sha256;\n    } else if (typeof define === 'function' && define.amd) {\n        define(function() { return sha256; }); \n    } else {\n        root.sha256 = sha256;\n    }\n})(this, function(exports) {\n\"use strict\";\nexports.__esModule = true;\n// SHA-256 (+ HMAC and PBKDF2) for JavaScript.\n//\n// Written in 2014-2016 by Dmitry Chestnykh.\n// Public domain, no warranty.\n//\n// Functions (accept and return Uint8Arrays):\n//\n//   sha256(message) -> hash\n//   sha256.hmac(key, message) -> mac\n//   sha256.pbkdf2(password, salt, rounds, dkLen) -> dk\n//\n//  Classes:\n//\n//   new sha256.Hash()\n//   new sha256.HMAC(key)\n//\nexports.digestLength = 32;\nexports.blockSize = 64;\n// SHA-256 constants\nvar K = new Uint32Array([\n    0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b,\n    0x59f111f1, 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01,\n    0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7,\n    0xc19bf174, 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc,\n    0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, 0x983e5152,\n    0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147,\n    0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc,\n    0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,\n    0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819,\n    0xd6990624, 0xf40e3585, 0x106aa070, 0x19a4c116, 0x1e376c08,\n    0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f,\n    0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208,\n    0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2\n]);\nfunction hashBlocks(w, v, p, pos, len) {\n    var a, b, c, d, e, f, g, h, u, i, j, t1, t2;\n    while (len >= 64) {\n        a = v[0];\n        b = v[1];\n        c = v[2];\n        d = v[3];\n        e = v[4];\n        f = v[5];\n        g = v[6];\n        h = v[7];\n        for (i = 0; i < 16; i++) {\n            j = pos + i * 4;\n            w[i] = (((p[j] & 0xff) << 24) | ((p[j + 1] & 0xff) << 16) |\n                ((p[j + 2] & 0xff) << 8) | (p[j + 3] & 0xff));\n        }\n        for (i = 16; i < 64; i++) {\n            u = w[i - 2];\n            t1 = (u >>> 17 | u << (32 - 17)) ^ (u >>> 19 | u << (32 - 19)) ^ (u >>> 10);\n            u = w[i - 15];\n            t2 = (u >>> 7 | u << (32 - 7)) ^ (u >>> 18 | u << (32 - 18)) ^ (u >>> 3);\n            w[i] = (t1 + w[i - 7] | 0) + (t2 + w[i - 16] | 0);\n        }\n        for (i = 0; i < 64; i++) {\n            t1 = (((((e >>> 6 | e << (32 - 6)) ^ (e >>> 11 | e << (32 - 11)) ^\n                (e >>> 25 | e << (32 - 25))) + ((e & f) ^ (~e & g))) | 0) +\n                ((h + ((K[i] + w[i]) | 0)) | 0)) | 0;\n            t2 = (((a >>> 2 | a << (32 - 2)) ^ (a >>> 13 | a << (32 - 13)) ^\n                (a >>> 22 | a << (32 - 22))) + ((a & b) ^ (a & c) ^ (b & c))) | 0;\n            h = g;\n            g = f;\n            f = e;\n            e = (d + t1) | 0;\n            d = c;\n            c = b;\n            b = a;\n            a = (t1 + t2) | 0;\n        }\n        v[0] += a;\n        v[1] += b;\n        v[2] += c;\n        v[3] += d;\n        v[4] += e;\n        v[5] += f;\n        v[6] += g;\n        v[7] += h;\n        pos += 64;\n        len -= 64;\n    }\n    return pos;\n}\n// Hash implements SHA256 hash algorithm.\nvar Hash = /** @class */ (function () {\n    function Hash() {\n        this.digestLength = exports.digestLength;\n        this.blockSize = exports.blockSize;\n        // Note: Int32Array is used instead of Uint32Array for performance reasons.\n        this.state = new Int32Array(8); // hash state\n        this.temp = new Int32Array(64); // temporary state\n        this.buffer = new Uint8Array(128); // buffer for data to hash\n        this.bufferLength = 0; // number of bytes in buffer\n        this.bytesHashed = 0; // number of total bytes hashed\n        this.finished = false; // indicates whether the hash was finalized\n        this.reset();\n    }\n    // Resets hash state making it possible\n    // to re-use this instance to hash other data.\n    Hash.prototype.reset = function () {\n        this.state[0] = 0x6a09e667;\n        this.state[1] = 0xbb67ae85;\n        this.state[2] = 0x3c6ef372;\n        this.state[3] = 0xa54ff53a;\n        this.state[4] = 0x510e527f;\n        this.state[5] = 0x9b05688c;\n        this.state[6] = 0x1f83d9ab;\n        this.state[7] = 0x5be0cd19;\n        this.bufferLength = 0;\n        this.bytesHashed = 0;\n        this.finished = false;\n        return this;\n    };\n    // Cleans internal buffers and re-initializes hash state.\n    Hash.prototype.clean = function () {\n        for (var i = 0; i < this.buffer.length; i++) {\n            this.buffer[i] = 0;\n        }\n        for (var i = 0; i < this.temp.length; i++) {\n            this.temp[i] = 0;\n        }\n        this.reset();\n    };\n    // Updates hash state with the given data.\n    //\n    // Optionally, length of the data can be specified to hash\n    // fewer bytes than data.length.\n    //\n    // Throws error when trying to update already finalized hash:\n    // instance must be reset to use it again.\n    Hash.prototype.update = function (data, dataLength) {\n        if (dataLength === void 0) { dataLength = data.length; }\n        if (this.finished) {\n            throw new Error(\"SHA256: can't update because hash was finished.\");\n        }\n        var dataPos = 0;\n        this.bytesHashed += dataLength;\n        if (this.bufferLength > 0) {\n            while (this.bufferLength < 64 && dataLength > 0) {\n                this.buffer[this.bufferLength++] = data[dataPos++];\n                dataLength--;\n            }\n            if (this.bufferLength === 64) {\n                hashBlocks(this.temp, this.state, this.buffer, 0, 64);\n                this.bufferLength = 0;\n            }\n        }\n        if (dataLength >= 64) {\n            dataPos = hashBlocks(this.temp, this.state, data, dataPos, dataLength);\n            dataLength %= 64;\n        }\n        while (dataLength > 0) {\n            this.buffer[this.bufferLength++] = data[dataPos++];\n            dataLength--;\n        }\n        return this;\n    };\n    // Finalizes hash state and puts hash into out.\n    //\n    // If hash was already finalized, puts the same value.\n    Hash.prototype.finish = function (out) {\n        if (!this.finished) {\n            var bytesHashed = this.bytesHashed;\n            var left = this.bufferLength;\n            var bitLenHi = (bytesHashed / 0x20000000) | 0;\n            var bitLenLo = bytesHashed << 3;\n            var padLength = (bytesHashed % 64 < 56) ? 64 : 128;\n            this.buffer[left] = 0x80;\n            for (var i = left + 1; i < padLength - 8; i++) {\n                this.buffer[i] = 0;\n            }\n            this.buffer[padLength - 8] = (bitLenHi >>> 24) & 0xff;\n            this.buffer[padLength - 7] = (bitLenHi >>> 16) & 0xff;\n            this.buffer[padLength - 6] = (bitLenHi >>> 8) & 0xff;\n            this.buffer[padLength - 5] = (bitLenHi >>> 0) & 0xff;\n            this.buffer[padLength - 4] = (bitLenLo >>> 24) & 0xff;\n            this.buffer[padLength - 3] = (bitLenLo >>> 16) & 0xff;\n            this.buffer[padLength - 2] = (bitLenLo >>> 8) & 0xff;\n            this.buffer[padLength - 1] = (bitLenLo >>> 0) & 0xff;\n            hashBlocks(this.temp, this.state, this.buffer, 0, padLength);\n            this.finished = true;\n        }\n        for (var i = 0; i < 8; i++) {\n            out[i * 4 + 0] = (this.state[i] >>> 24) & 0xff;\n            out[i * 4 + 1] = (this.state[i] >>> 16) & 0xff;\n            out[i * 4 + 2] = (this.state[i] >>> 8) & 0xff;\n            out[i * 4 + 3] = (this.state[i] >>> 0) & 0xff;\n        }\n        return this;\n    };\n    // Returns the final hash digest.\n    Hash.prototype.digest = function () {\n        var out = new Uint8Array(this.digestLength);\n        this.finish(out);\n        return out;\n    };\n    // Internal function for use in HMAC for optimization.\n    Hash.prototype._saveState = function (out) {\n        for (var i = 0; i < this.state.length; i++) {\n            out[i] = this.state[i];\n        }\n    };\n    // Internal function for use in HMAC for optimization.\n    Hash.prototype._restoreState = function (from, bytesHashed) {\n        for (var i = 0; i < this.state.length; i++) {\n            this.state[i] = from[i];\n        }\n        this.bytesHashed = bytesHashed;\n        this.finished = false;\n        this.bufferLength = 0;\n    };\n    return Hash;\n}());\nexports.Hash = Hash;\n// HMAC implements HMAC-SHA256 message authentication algorithm.\nvar HMAC = /** @class */ (function () {\n    function HMAC(key) {\n        this.inner = new Hash();\n        this.outer = new Hash();\n        this.blockSize = this.inner.blockSize;\n        this.digestLength = this.inner.digestLength;\n        var pad = new Uint8Array(this.blockSize);\n        if (key.length > this.blockSize) {\n            (new Hash()).update(key).finish(pad).clean();\n        }\n        else {\n            for (var i = 0; i < key.length; i++) {\n                pad[i] = key[i];\n            }\n        }\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] ^= 0x36;\n        }\n        this.inner.update(pad);\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] ^= 0x36 ^ 0x5c;\n        }\n        this.outer.update(pad);\n        this.istate = new Uint32Array(8);\n        this.ostate = new Uint32Array(8);\n        this.inner._saveState(this.istate);\n        this.outer._saveState(this.ostate);\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] = 0;\n        }\n    }\n    // Returns HMAC state to the state initialized with key\n    // to make it possible to run HMAC over the other data with the same\n    // key without creating a new instance.\n    HMAC.prototype.reset = function () {\n        this.inner._restoreState(this.istate, this.inner.blockSize);\n        this.outer._restoreState(this.ostate, this.outer.blockSize);\n        return this;\n    };\n    // Cleans HMAC state.\n    HMAC.prototype.clean = function () {\n        for (var i = 0; i < this.istate.length; i++) {\n            this.ostate[i] = this.istate[i] = 0;\n        }\n        this.inner.clean();\n        this.outer.clean();\n    };\n    // Updates state with provided data.\n    HMAC.prototype.update = function (data) {\n        this.inner.update(data);\n        return this;\n    };\n    // Finalizes HMAC and puts the result in out.\n    HMAC.prototype.finish = function (out) {\n        if (this.outer.finished) {\n            this.outer.finish(out);\n        }\n        else {\n            this.inner.finish(out);\n            this.outer.update(out, this.digestLength).finish(out);\n        }\n        return this;\n    };\n    // Returns message authentication code.\n    HMAC.prototype.digest = function () {\n        var out = new Uint8Array(this.digestLength);\n        this.finish(out);\n        return out;\n    };\n    return HMAC;\n}());\nexports.HMAC = HMAC;\n// Returns SHA256 hash of data.\nfunction hash(data) {\n    var h = (new Hash()).update(data);\n    var digest = h.digest();\n    h.clean();\n    return digest;\n}\nexports.hash = hash;\n// Function hash is both available as module.hash and as default export.\nexports[\"default\"] = hash;\n// Returns HMAC-SHA256 of data under the key.\nfunction hmac(key, data) {\n    var h = (new HMAC(key)).update(data);\n    var digest = h.digest();\n    h.clean();\n    return digest;\n}\nexports.hmac = hmac;\n// Fills hkdf buffer like this:\n// T(1) = HMAC-Hash(PRK, T(0) | info | 0x01)\nfunction fillBuffer(buffer, hmac, info, counter) {\n    // Counter is a byte value: check if it overflowed.\n    var num = counter[0];\n    if (num === 0) {\n        throw new Error(\"hkdf: cannot expand more\");\n    }\n    // Prepare HMAC instance for new data with old key.\n    hmac.reset();\n    // Hash in previous output if it was generated\n    // (i.e. counter is greater than 1).\n    if (num > 1) {\n        hmac.update(buffer);\n    }\n    // Hash in info if it exists.\n    if (info) {\n        hmac.update(info);\n    }\n    // Hash in the counter.\n    hmac.update(counter);\n    // Output result to buffer and clean HMAC instance.\n    hmac.finish(buffer);\n    // Increment counter inside typed array, this works properly.\n    counter[0]++;\n}\nvar hkdfSalt = new Uint8Array(exports.digestLength); // Filled with zeroes.\nfunction hkdf(key, salt, info, length) {\n    if (salt === void 0) { salt = hkdfSalt; }\n    if (length === void 0) { length = 32; }\n    var counter = new Uint8Array([1]);\n    // HKDF-Extract uses salt as HMAC key, and key as data.\n    var okm = hmac(salt, key);\n    // Initialize HMAC for expanding with extracted key.\n    // Ensure no collisions with `hmac` function.\n    var hmac_ = new HMAC(okm);\n    // Allocate buffer.\n    var buffer = new Uint8Array(hmac_.digestLength);\n    var bufpos = buffer.length;\n    var out = new Uint8Array(length);\n    for (var i = 0; i < length; i++) {\n        if (bufpos === buffer.length) {\n            fillBuffer(buffer, hmac_, info, counter);\n            bufpos = 0;\n        }\n        out[i] = buffer[bufpos++];\n    }\n    hmac_.clean();\n    buffer.fill(0);\n    counter.fill(0);\n    return out;\n}\nexports.hkdf = hkdf;\n// Derives a key from password and salt using PBKDF2-HMAC-SHA256\n// with the given number of iterations.\n//\n// The number of bytes returned is equal to dkLen.\n//\n// (For better security, avoid dkLen greater than hash length - 32 bytes).\nfunction pbkdf2(password, salt, iterations, dkLen) {\n    var prf = new HMAC(password);\n    var len = prf.digestLength;\n    var ctr = new Uint8Array(4);\n    var t = new Uint8Array(len);\n    var u = new Uint8Array(len);\n    var dk = new Uint8Array(dkLen);\n    for (var i = 0; i * len < dkLen; i++) {\n        var c = i + 1;\n        ctr[0] = (c >>> 24) & 0xff;\n        ctr[1] = (c >>> 16) & 0xff;\n        ctr[2] = (c >>> 8) & 0xff;\n        ctr[3] = (c >>> 0) & 0xff;\n        prf.reset();\n        prf.update(salt);\n        prf.update(ctr);\n        prf.finish(u);\n        for (var j = 0; j < len; j++) {\n            t[j] = u[j];\n        }\n        for (var j = 2; j <= iterations; j++) {\n            prf.reset();\n            prf.update(u).finish(u);\n            for (var k = 0; k < len; k++) {\n                t[k] ^= u[k];\n            }\n        }\n        for (var j = 0; j < len && i * len + j < dkLen; j++) {\n            dk[i * len + j] = t[j];\n        }\n    }\n    for (var i = 0; i < len; i++) {\n        t[i] = u[i] = 0;\n    }\n    for (var i = 0; i < 4; i++) {\n        ctr[i] = 0;\n    }\n    prf.clean();\n    return dk;\n}\nexports.pbkdf2 = pbkdf2;\n});\n","\"use strict\";\nconst peq = new Uint32Array(0x10000);\nconst myers_32 = (a, b) => {\n  const n = a.length;\n  const m = b.length;\n  const lst = 1 << (n - 1);\n  let pv = -1;\n  let mv = 0;\n  let sc = n;\n  let i = n;\n  while (i--) {\n    peq[a.charCodeAt(i)] |= 1 << i;\n  }\n  for (i = 0; i < m; i++) {\n    let eq = peq[b.charCodeAt(i)];\n    const xv = eq | mv;\n    eq |= ((eq & pv) + pv) ^ pv;\n    mv |= ~(eq | pv);\n    pv &= eq;\n    if (mv & lst) {\n      sc++;\n    }\n    if (pv & lst) {\n      sc--;\n    }\n    mv = (mv << 1) | 1;\n    pv = (pv << 1) | ~(xv | mv);\n    mv &= xv;\n  }\n  i = n;\n  while (i--) {\n    peq[a.charCodeAt(i)] = 0;\n  }\n  return sc;\n};\n\nconst myers_x = (a, b) => {\n  const n = a.length;\n  const m = b.length;\n  const mhc = [];\n  const phc = [];\n  const hsize = Math.ceil(n / 32);\n  const vsize = Math.ceil(m / 32);\n  let score = m;\n  for (let i = 0; i < hsize; i++) {\n    phc[i] = -1;\n    mhc[i] = 0;\n  }\n  let j = 0;\n  for (; j < vsize - 1; j++) {\n    let mv = 0;\n    let pv = -1;\n    const start = j * 32;\n    const end = Math.min(32, m) + start;\n    for (let k = start; k < end; k++) {\n      peq[b.charCodeAt(k)] |= 1 << k;\n    }\n    score = m;\n    for (let i = 0; i < n; i++) {\n      const eq = peq[a.charCodeAt(i)];\n      const pb = (phc[(i / 32) | 0] >>> i) & 1;\n      const mb = (mhc[(i / 32) | 0] >>> i) & 1;\n      const xv = eq | mv;\n      const xh = ((((eq | mb) & pv) + pv) ^ pv) | eq | mb;\n      let ph = mv | ~(xh | pv);\n      let mh = pv & xh;\n      if ((ph >>> 31) ^ pb) {\n        phc[(i / 32) | 0] ^= 1 << i;\n      }\n      if ((mh >>> 31) ^ mb) {\n        mhc[(i / 32) | 0] ^= 1 << i;\n      }\n      ph = (ph << 1) | pb;\n      mh = (mh << 1) | mb;\n      pv = mh | ~(xv | ph);\n      mv = ph & xv;\n    }\n    for (let k = start; k < end; k++) {\n      peq[b.charCodeAt(k)] = 0;\n    }\n  }\n  let mv = 0;\n  let pv = -1;\n  const start = j * 32;\n  const end = Math.min(32, m - start) + start;\n  for (let k = start; k < end; k++) {\n    peq[b.charCodeAt(k)] |= 1 << k;\n  }\n  score = m;\n  for (let i = 0; i < n; i++) {\n    const eq = peq[a.charCodeAt(i)];\n    const pb = (phc[(i / 32) | 0] >>> i) & 1;\n    const mb = (mhc[(i / 32) | 0] >>> i) & 1;\n    const xv = eq | mv;\n    const xh = ((((eq | mb) & pv) + pv) ^ pv) | eq | mb;\n    let ph = mv | ~(xh | pv);\n    let mh = pv & xh;\n    score += (ph >>> (m - 1)) & 1;\n    score -= (mh >>> (m - 1)) & 1;\n    if ((ph >>> 31) ^ pb) {\n      phc[(i / 32) | 0] ^= 1 << i;\n    }\n    if ((mh >>> 31) ^ mb) {\n      mhc[(i / 32) | 0] ^= 1 << i;\n    }\n    ph = (ph << 1) | pb;\n    mh = (mh << 1) | mb;\n    pv = mh | ~(xv | ph);\n    mv = ph & xv;\n  }\n  for (let k = start; k < end; k++) {\n    peq[b.charCodeAt(k)] = 0;\n  }\n  return score;\n};\n\nconst distance = (a, b) => {\n  if (a.length > b.length) {\n    const tmp = b;\n    b = a;\n    a = tmp;\n  }\n  if (a.length === 0) {\n    return b.length;\n  }\n  if (a.length <= 32) {\n    return myers_32(a, b);\n  }\n  return myers_x(a, b);\n};\n\nconst closest = (str, arr) => {\n  let min_distance = Infinity;\n  let min_index = 0;\n  for (let i = 0; i < arr.length; i++) {\n    const dist = distance(str, arr[i]);\n    if (dist < min_distance) {\n      min_distance = dist;\n      min_index = i;\n    }\n  }\n  return arr[min_index];\n};\n\nmodule.exports = {\n  closest, distance\n}\n","\"use strict\";\n// Reference: https://www.geeksforgeeks.org/jaro-and-jaro-winkler-similarity/\nObject.defineProperty(exports, \"__esModule\", { value: true });\nexports.jaroWinkler = exports.jaro = void 0;\n/**\n *\n * @param str1 String 1 for compare\n * @param str2 String 2 for compare\n * @param options to control case sensitive or not\n */\nfunction jaro(str1, str2, options) {\n    // Exit early if either are empty.\n    if (str1.length === 0 || str2.length === 0) {\n        return 0;\n    }\n    // Convert to upper if case-sensitive is false.\n    if (options && !options.caseSensitive) {\n        str1 = str1.toUpperCase();\n        str2 = str2.toUpperCase();\n    }\n    // Exact match\n    if (str1 === str2) {\n        return 1;\n    }\n    // Number of matches\n    var m = 0;\n    // Length of two Strings\n    var len1 = str1.length;\n    var len2 = str2.length;\n    // Maximum distance\n    var window = Math.floor(Math.max(len1, len2) / 2) - 1;\n    // Hash for matches\n    var str1Hash = new Array(len1);\n    var str2Hash = new Array(len2);\n    for (var i = 0; i < len1; i++) {\n        for (var j = Math.max(0, i - window); j <= Math.min(len2, i + window + 1); j++) {\n            if (!str1Hash[i] && !str2Hash[j] && str1[i] === str2[j]) {\n                ++m;\n                str1Hash[i] = str2Hash[j] = true;\n                break;\n            }\n        }\n    }\n    // Exit early if no matches were found.\n    if (m === 0) {\n        return 0;\n    }\n    // Count the transpositions.\n    var t = 0;\n    var point = 0;\n    for (var i = 0; i < len1; i++) {\n        if (str1Hash[i]) {\n            while (!str2Hash[point]) {\n                point++;\n            }\n            if (str1.charAt(i) !== str2.charAt(point++)) {\n                t++;\n            }\n        }\n    }\n    t /= 2;\n    return (m / len1 + m / len2 + (m - t) / m) / 3;\n}\nexports.jaro = jaro;\n/**\n *\n * @param str1 String 1 for compare\n * @param str2 String 2 for compare\n * @param options to control case sensitive or not\n */\nfunction jaroWinkler(str1, str2, options) {\n    // Jaro Distance\n    var jaroDist = jaro(str1, str2, options);\n    // Same prefix length, maxium is 4\n    var prefix = 0;\n    if (jaroDist > 0.7) {\n        var minIndex = Math.min(str1.length, str2.length);\n        var i = 0;\n        while (str1[i] === str2[i] && i < 4 && i < minIndex) {\n            ++prefix;\n            i++;\n        }\n        jaroDist += 0.1 * prefix * (1 - jaroDist);\n    }\n    return jaroDist;\n}\nexports.jaroWinkler = jaroWinkler;\n","export const jsonSdfMonomerLibDict = {\n    'monomerType': null,\n    'smiles': null,\n    'name': 'MonomerName',\n    'author': null,\n    'molfile': 'molecule',\n    'naturalAnalog': 'MonomerNaturalAnalogCode',\n    'rgroups': 'MonomerCaps',\n    'createDate': null,\n    'id': null,\n    'polymerType': 'MonomerType',\n    'symbol': 'MonomerCode'\n};\nexport const RGROUP_FIELD = 'rgroups';\nexport const CAP_GROUP_SMILES = 'capGroupSmiles';\nexport const RGROUP_ALTER_ID = 'alternateId';\nexport const CAP_GROUP_NAME = 'capGroupName';\nexport const RGROUP_LABEL = 'label';\nexport const MONOMER_SYMBOL = 'symbol';\nexport const SDF_MONOMER_NAME = 'MonomerName';\n// range of hex nubers used in PepSea library to endode monomers\nexport const MONOMER_ENCODE_MIN = 0x100;\nexport const MONOMER_ENCODE_MAX = 0x40A;\n","class Cell {\n    constructor() {\n        this.value = 0;\n        this.parentI = 0;\n        this.parentJ = 0;\n    }\n}\n;\nexport class Aligned {\n    constructor(seq1_, seq2_, score_) {\n        this.seq1 = seq1_;\n        this.seq2 = seq2_;\n        this.score = score_;\n    }\n}\n;\nexport class SequenceAlignment {\n    constructor(seq1_, seq2_, gap_, alignGrid_ = \"\") {\n        this.seq1 = '';\n        this.seq2 = '';\n        this.gap = 1;\n        this.len1 = 0;\n        this.len2 = 0;\n        this.step = 1;\n        this.alignGrid = [];\n        this.scores = [];\n        this.dpGrid = [];\n        this.connections = {};\n        this.BLOSUM45 = [[5, -2, -1, -2, -1, -1, -1, 0, -2, -1, -1, -1, -1, -2, -1, 1, 0, -2, -2, 0, -1, -1, 0, -5],\n            [-2, 7, 0, -1, -3, 1, 0, -2, 0, -3, -2, 3, -1, -2, -2, -1, -1, -2, -1, -2, -1, 0, -1, -5],\n            [-1, 0, 6, 2, -2, 0, 0, 0, 1, -2, -3, 0, -2, -2, -2, 1, 0, -4, -2, -3, 4, 0, -1, -5],\n            [-2, -1, 2, 7, -3, 0, 2, -1, 0, -4, -3, 0, -3, -4, -1, 0, -1, -4, -2, -3, 5, 1, -1, -5],\n            [-1, -3, -2, -3, 12, -3, -3, -3, -3, -3, -2, -3, -2, -2, -4, -1, -1, -5, -3, -1, -2, -3, -2, -5],\n            [-1, 1, 0, 0, -3, 6, 2, -2, 1, -2, -2, 1, 0, -4, -1, 0, -1, -2, -1, -3, 0, 4, -1, -5],\n            [-1, 0, 0, 2, -3, 2, 6, -2, 0, -3, -2, 1, -2, -3, 0, 0, -1, -3, -2, -3, 1, 4, -1, -5],\n            [0, -2, 0, -1, -3, -2, -2, 7, -2, -4, -3, -2, -2, -3, -2, 0, -2, -2, -3, -3, -1, -2, -1, -5],\n            [-2, 0, 1, 0, -3, 1, 0, -2, 10, -3, -2, -1, 0, -2, -2, -1, -2, -3, 2, -3, 0, 0, -1, -5],\n            [-1, -3, -2, -4, -3, -2, -3, -4, -3, 5, 2, -3, 2, 0, -2, -2, -1, -2, 0, 3, -3, -3, -1, -5],\n            [-1, -2, -3, -3, -2, -2, -2, -3, -2, 2, 5, -3, 2, 1, -3, -3, -1, -2, 0, 1, -3, -2, -1, -5],\n            [-1, 3, 0, 0, -3, 1, 1, -2, -1, -3, -3, 5, -1, -3, -1, -1, -1, -2, -1, -2, 0, 1, -1, -5],\n            [-1, -1, -2, -3, -2, 0, -2, -2, 0, 2, 2, -1, 6, 0, -2, -2, -1, -2, 0, 1, -2, -1, -1, -5],\n            [-2, -2, -2, -4, -2, -4, -3, -3, -2, 0, 1, -3, 0, 8, -3, -2, -1, 1, 3, 0, -3, -3, -1, -5],\n            [-1, -2, -2, -1, -4, -1, 0, -2, -2, -2, -3, -1, -2, -3, 9, -1, -1, -3, -3, -3, -2, -1, -1, -5],\n            [1, -1, 1, 0, -1, 0, 0, 0, -1, -2, -3, -1, -2, -2, -1, 4, 2, -4, -2, -1, 0, 0, 0, -5],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -1, -1, 2, 5, -3, -1, 0, 0, -1, 0, -5],\n            [-2, -2, -4, -4, -5, -2, -3, -2, -3, -2, -2, -2, -2, 1, -3, -4, -3, 15, 3, -3, -4, -2, -2, -5],\n            [-2, -1, -2, -2, -3, -1, -2, -3, 2, 0, 0, -1, 0, 3, -3, -2, -1, 3, 8, -1, -2, -2, -1, -5],\n            [0, -2, -3, -3, -1, -3, -3, -3, -3, 3, 1, -2, 1, 0, -3, -1, 0, -3, -1, 5, -3, -3, -1, -5],\n            [-1, -1, 4, 5, -2, 0, 1, -1, 0, -3, -3, 0, -2, -3, -2, 0, 0, -4, -2, -3, 4, 2, -1, -5],\n            [-1, 0, 0, 1, -3, 4, 4, -2, 0, -3, -2, 1, -1, -3, -1, 0, -1, -2, -2, -3, 2, 4, -1, -5],\n            [0, -1, -1, -1, -2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 0, -2, -1, -1, -1, -1, -1, -5],\n            [-5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, 1]];\n        this.BLOSUM50 = [[5, -2, -1, -2, -1, -1, -1, 0, -2, -1, -2, -1, -1, -3, -1, 1, 0, -3, -2, 0, -2, -1, -1, -5],\n            [-2, 7, -1, -2, -4, 1, 0, -3, 0, -4, -3, 3, -2, -3, -3, -1, -1, -3, -1, -3, -1, 0, -1, -5],\n            [-1, -1, 7, 2, -2, 0, 0, 0, 1, -3, -4, 0, -2, -4, -2, 1, 0, -4, -2, -3, 4, 0, -1, -5],\n            [-2, -2, 2, 8, -4, 0, 2, -1, -1, -4, -4, -1, -4, -5, -1, 0, -1, -5, -3, -4, 5, 1, -1, -5],\n            [-1, -4, -2, -4, 13, -3, -3, -3, -3, -2, -2, -3, -2, -2, -4, -1, -1, -5, -3, -1, -3, -3, -2, -5],\n            [-1, 1, 0, 0, -3, 7, 2, -2, 1, -3, -2, 2, 0, -4, -1, 0, -1, -1, -1, -3, 0, 4, -1, -5],\n            [-1, 0, 0, 2, -3, 2, 6, -3, 0, -4, -3, 1, -2, -3, -1, -1, -1, -3, -2, -3, 1, 5, -1, -5],\n            [0, -3, 0, -1, -3, -2, -3, 8, -2, -4, -4, -2, -3, -4, -2, 0, -2, -3, -3, -4, -1, -2, -2, -5],\n            [-2, 0, 1, -1, -3, 1, 0, -2, 10, -4, -3, 0, -1, -1, -2, -1, -2, -3, 2, -4, 0, 0, -1, -5],\n            [-1, -4, -3, -4, -2, -3, -4, -4, -4, 5, 2, -3, 2, 0, -3, -3, -1, -3, -1, 4, -4, -3, -1, -5],\n            [-2, -3, -4, -4, -2, -2, -3, -4, -3, 2, 5, -3, 3, 1, -4, -3, -1, -2, -1, 1, -4, -3, -1, -5],\n            [-1, 3, 0, -1, -3, 2, 1, -2, 0, -3, -3, 6, -2, -4, -1, 0, -1, -3, -2, -3, 0, 1, -1, -5],\n            [-1, -2, -2, -4, -2, 0, -2, -3, -1, 2, 3, -2, 7, 0, -3, -2, -1, -1, 0, 1, -3, -1, -1, -5],\n            [-3, -3, -4, -5, -2, -4, -3, -4, -1, 0, 1, -4, 0, 8, -4, -3, -2, 1, 4, -1, -4, -4, -2, -5],\n            [-1, -3, -2, -1, -4, -1, -1, -2, -2, -3, -4, -1, -3, -4, 10, -1, -1, -4, -3, -3, -2, -1, -2, -5],\n            [1, -1, 1, 0, -1, 0, -1, 0, -1, -3, -3, 0, -2, -3, -1, 5, 2, -4, -2, -2, 0, 0, -1, -5],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -2, -1, 2, 5, -3, -2, 0, 0, -1, 0, -5],\n            [-3, -3, -4, -5, -5, -1, -3, -3, -3, -3, -2, -3, -1, 1, -4, -4, -3, 15, 2, -3, -5, -2, -3, -5],\n            [-2, -1, -2, -3, -3, -1, -2, -3, 2, -1, -1, -2, 0, 4, -3, -2, -2, 2, 8, -1, -3, -2, -1, -5],\n            [0, -3, -3, -4, -1, -3, -3, -4, -4, 4, 1, -3, 1, -1, -3, -2, 0, -3, -1, 5, -4, -3, -1, -5],\n            [-2, -1, 4, 5, -3, 0, 1, -1, 0, -4, -4, 0, -3, -4, -2, 0, 0, -5, -3, -4, 5, 2, -1, -5],\n            [-1, 0, 0, 1, -3, 4, 5, -2, 0, -3, -3, 1, -1, -4, -1, 0, -1, -2, -2, -3, 2, 5, -1, -5],\n            [-1, -1, -1, -1, -2, -1, -1, -2, -1, -1, -1, -1, -1, -2, -2, -1, 0, -3, -1, -1, -1, -1, -1, -5],\n            [-5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, 1]];\n        this.BLOSUM62 = [[4, -1, -2, -2, 0, -1, -1, 0, -2, -1, -1, -1, -1, -2, -1, 1, 0, -3, -2, 0, -2, -1, 0, -4],\n            [-1, 5, 0, -2, -3, 1, 0, -2, 0, -3, -2, 2, -1, -3, -2, -1, -1, -3, -2, -3, -1, 0, -1, -4],\n            [-2, 0, 6, 1, -3, 0, 0, 0, 1, -3, -3, 0, -2, -3, -2, 1, 0, -4, -2, -3, 3, 0, -1, -4],\n            [-2, -2, 1, 6, -3, 0, 2, -1, -1, -3, -4, -1, -3, -3, -1, 0, -1, -4, -3, -3, 4, 1, -1, -4],\n            [0, -3, -3, -3, 9, -3, -4, -3, -3, -1, -1, -3, -1, -2, -3, -1, -1, -2, -2, -1, -3, -3, -2, -4],\n            [-1, 1, 0, 0, -3, 5, 2, -2, 0, -3, -2, 1, 0, -3, -1, 0, -1, -2, -1, -2, 0, 3, -1, -4],\n            [-1, 0, 0, 2, -4, 2, 5, -2, 0, -3, -3, 1, -2, -3, -1, 0, -1, -3, -2, -2, 1, 4, -1, -4],\n            [0, -2, 0, -1, -3, -2, -2, 6, -2, -4, -4, -2, -3, -3, -2, 0, -2, -2, -3, -3, -1, -2, -1, -4],\n            [-2, 0, 1, -1, -3, 0, 0, -2, 8, -3, -3, -1, -2, -1, -2, -1, -2, -2, 2, -3, 0, 0, -1, -4],\n            [-1, -3, -3, -3, -1, -3, -3, -4, -3, 4, 2, -3, 1, 0, -3, -2, -1, -3, -1, 3, -3, -3, -1, -4],\n            [-1, -2, -3, -4, -1, -2, -3, -4, -3, 2, 4, -2, 2, 0, -3, -2, -1, -2, -1, 1, -4, -3, -1, -4],\n            [-1, 2, 0, -1, -3, 1, 1, -2, -1, -3, -2, 5, -1, -3, -1, 0, -1, -3, -2, -2, 0, 1, -1, -4],\n            [-1, -1, -2, -3, -1, 0, -2, -3, -2, 1, 2, -1, 5, 0, -2, -1, -1, -1, -1, 1, -3, -1, -1, -4],\n            [-2, -3, -3, -3, -2, -3, -3, -3, -1, 0, 0, -3, 0, 6, -4, -2, -2, 1, 3, -1, -3, -3, -1, -4],\n            [-1, -2, -2, -1, -3, -1, -1, -2, -2, -3, -3, -1, -2, -4, 7, -1, -1, -4, -3, -2, -2, -1, -2, -4],\n            [1, -1, 1, 0, -1, 0, 0, 0, -1, -2, -2, 0, -1, -2, -1, 4, 1, -3, -2, -2, 0, 0, 0, -4],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -2, -1, 1, 5, -2, -2, 0, -1, -1, 0, -4],\n            [-3, -3, -4, -4, -2, -2, -3, -2, -2, -3, -2, -3, -1, 1, -4, -3, -2, 11, 2, -3, -4, -3, -2, -4],\n            [-2, -2, -2, -3, -2, -1, -2, -3, 2, -1, -1, -2, -1, 3, -3, -2, -2, 2, 7, -1, -3, -2, -1, -4],\n            [0, -3, -3, -3, -1, -2, -2, -3, -3, 3, 1, -2, 1, -1, -2, -2, 0, -3, -1, 4, -3, -2, -1, -4],\n            [-2, -1, 3, 4, -3, 0, 1, -1, 0, -3, -4, 0, -3, -3, -2, 0, -1, -4, -3, -3, 4, 1, -1, -4],\n            [-1, 0, 0, 1, -3, 3, 4, -2, 0, -3, -3, 1, -1, -3, -1, 0, -1, -3, -2, -2, 1, 4, -1, -4],\n            [0, -1, -1, -1, -2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -2, 0, 0, -2, -1, -1, -1, -1, -1, -4],\n            [-4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, 1]];\n        this.BLOSUM80 = [[7, -3, -3, -3, -1, -2, -2, 0, -3, -3, -3, -1, -2, -4, -1, 2, 0, -5, -4, -1, -3, -2, -1, -8],\n            [-3, 9, -1, -3, -6, 1, -1, -4, 0, -5, -4, 3, -3, -5, -3, -2, -2, -5, -4, -4, -2, 0, -2, -8],\n            [-3, -1, 9, 2, -5, 0, -1, -1, 1, -6, -6, 0, -4, -6, -4, 1, 0, -7, -4, -5, 5, -1, -2, -8],\n            [-3, -3, 2, 10, -7, -1, 2, -3, -2, -7, -7, -2, -6, -6, -3, -1, -2, -8, -6, -6, 6, 1, -3, -8],\n            [-1, -6, -5, -7, 13, -5, -7, -6, -7, -2, -3, -6, -3, -4, -6, -2, -2, -5, -5, -2, -6, -7, -4, -8],\n            [-2, 1, 0, -1, -5, 9, 3, -4, 1, -5, -4, 2, -1, -5, -3, -1, -1, -4, -3, -4, -1, 5, -2, -8],\n            [-2, -1, -1, 2, -7, 3, 8, -4, 0, -6, -6, 1, -4, -6, -2, -1, -2, -6, -5, -4, 1, 6, -2, -8],\n            [0, -4, -1, -3, -6, -4, -4, 9, -4, -7, -7, -3, -5, -6, -5, -1, -3, -6, -6, -6, -2, -4, -3, -8],\n            [-3, 0, 1, -2, -7, 1, 0, -4, 12, -6, -5, -1, -4, -2, -4, -2, -3, -4, 3, -5, -1, 0, -2, -8],\n            [-3, -5, -6, -7, -2, -5, -6, -7, -6, 7, 2, -5, 2, -1, -5, -4, -2, -5, -3, 4, -6, -6, -2, -8],\n            [-3, -4, -6, -7, -3, -4, -6, -7, -5, 2, 6, -4, 3, 0, -5, -4, -3, -4, -2, 1, -7, -5, -2, -8],\n            [-1, 3, 0, -2, -6, 2, 1, -3, -1, -5, -4, 8, -3, -5, -2, -1, -1, -6, -4, -4, -1, 1, -2, -8],\n            [-2, -3, -4, -6, -3, -1, -4, -5, -4, 2, 3, -3, 9, 0, -4, -3, -1, -3, -3, 1, -5, -3, -2, -8],\n            [-4, -5, -6, -6, -4, -5, -6, -6, -2, -1, 0, -5, 0, 10, -6, -4, -4, 0, 4, -2, -6, -6, -3, -8],\n            [-1, -3, -4, -3, -6, -3, -2, -5, -4, -5, -5, -2, -4, -6, 12, -2, -3, -7, -6, -4, -4, -2, -3, -8],\n            [2, -2, 1, -1, -2, -1, -1, -1, -2, -4, -4, -1, -3, -4, -2, 7, 2, -6, -3, -3, 0, -1, -1, -8],\n            [0, -2, 0, -2, -2, -1, -2, -3, -3, -2, -3, -1, -1, -4, -3, 2, 8, -5, -3, 0, -1, -2, -1, -8],\n            [-5, -5, -7, -8, -5, -4, -6, -6, -4, -5, -4, -6, -3, 0, -7, -6, -5, 16, 3, -5, -8, -5, -5, -8],\n            [-4, -4, -4, -6, -5, -3, -5, -6, 3, -3, -2, -4, -3, 4, -6, -3, -3, 3, 11, -3, -5, -4, -3, -8],\n            [-1, -4, -5, -6, -2, -4, -4, -6, -5, 4, 1, -4, 1, -2, -4, -3, 0, -5, -3, 7, -6, -4, -2, -8],\n            [-3, -2, 5, 6, -6, -1, 1, -2, -1, -6, -7, -1, -5, -6, -4, 0, -1, -8, -5, -6, 6, 0, -3, -8],\n            [-2, 0, -1, 1, -7, 5, 6, -4, 0, -6, -5, 1, -3, -6, -2, -1, -2, -5, -4, -4, 0, 6, -1, -8],\n            [-1, -2, -2, -3, -4, -2, -2, -3, -2, -2, -2, -2, -2, -3, -3, -1, -1, -5, -3, -2, -3, -1, -2, -8],\n            [-8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, 1]];\n        this.BLOSUM90 = [[5, -2, -2, -3, -1, -1, -1, 0, -2, -2, -2, -1, -2, -3, -1, 1, 0, -4, -3, -1, -2, -1, -1, -6],\n            [-2, 6, -1, -3, -5, 1, -1, -3, 0, -4, -3, 2, -2, -4, -3, -1, -2, -4, -3, -3, -2, 0, -2, -6],\n            [-2, -1, 7, 1, -4, 0, -1, -1, 0, -4, -4, 0, -3, -4, -3, 0, 0, -5, -3, -4, 4, -1, -2, -6],\n            [-3, -3, 1, 7, -5, -1, 1, -2, -2, -5, -5, -1, -4, -5, -3, -1, -2, -6, -4, -5, 4, 0, -2, -6],\n            [-1, -5, -4, -5, 9, -4, -6, -4, -5, -2, -2, -4, -2, -3, -4, -2, -2, -4, -4, -2, -4, -5, -3, -6],\n            [-1, 1, 0, -1, -4, 7, 2, -3, 1, -4, -3, 1, 0, -4, -2, -1, -1, -3, -3, -3, -1, 4, -1, -6],\n            [-1, -1, -1, 1, -6, 2, 6, -3, -1, -4, -4, 0, -3, -5, -2, -1, -1, -5, -4, -3, 0, 4, -2, -6],\n            [0, -3, -1, -2, -4, -3, -3, 6, -3, -5, -5, -2, -4, -5, -3, -1, -3, -4, -5, -5, -2, -3, -2, -6],\n            [-2, 0, 0, -2, -5, 1, -1, -3, 8, -4, -4, -1, -3, -2, -3, -2, -2, -3, 1, -4, -1, 0, -2, -6],\n            [-2, -4, -4, -5, -2, -4, -4, -5, -4, 5, 1, -4, 1, -1, -4, -3, -1, -4, -2, 3, -5, -4, -2, -6],\n            [-2, -3, -4, -5, -2, -3, -4, -5, -4, 1, 5, -3, 2, 0, -4, -3, -2, -3, -2, 0, -5, -4, -2, -6],\n            [-1, 2, 0, -1, -4, 1, 0, -2, -1, -4, -3, 6, -2, -4, -2, -1, -1, -5, -3, -3, -1, 1, -1, -6],\n            [-2, -2, -3, -4, -2, 0, -3, -4, -3, 1, 2, -2, 7, -1, -3, -2, -1, -2, -2, 0, -4, -2, -1, -6],\n            [-3, -4, -4, -5, -3, -4, -5, -5, -2, -1, 0, -4, -1, 7, -4, -3, -3, 0, 3, -2, -4, -4, -2, -6],\n            [-1, -3, -3, -3, -4, -2, -2, -3, -3, -4, -4, -2, -3, -4, 8, -2, -2, -5, -4, -3, -3, -2, -2, -6],\n            [1, -1, 0, -1, -2, -1, -1, -1, -2, -3, -3, -1, -2, -3, -2, 5, 1, -4, -3, -2, 0, -1, -1, -6],\n            [0, -2, 0, -2, -2, -1, -1, -3, -2, -1, -2, -1, -1, -3, -2, 1, 6, -4, -2, -1, -1, -1, -1, -6],\n            [-4, -4, -5, -6, -4, -3, -5, -4, -3, -4, -3, -5, -2, 0, -5, -4, -4, 11, 2, -3, -6, -4, -3, -6],\n            [-3, -3, -3, -4, -4, -3, -4, -5, 1, -2, -2, -3, -2, 3, -4, -3, -2, 2, 8, -3, -4, -3, -2, -6],\n            [-1, -3, -4, -5, -2, -3, -3, -5, -4, 3, 0, -3, 0, -2, -3, -2, -1, -3, -3, 5, -4, -3, -2, -6],\n            [-2, -2, 4, 4, -4, -1, 0, -2, -1, -5, -5, -1, -4, -4, -3, 0, -1, -6, -4, -4, 4, 0, -2, -6],\n            [-1, 0, -1, 0, -5, 4, 4, -3, 0, -4, -4, 1, -2, -4, -2, -1, -1, -4, -3, -3, 0, 4, -1, -6],\n            [-1, -2, -2, -2, -3, -1, -2, -2, -2, -2, -2, -1, -1, -2, -2, -1, -1, -3, -2, -2, -2, -1, -2, -6],\n            [-6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, 1]];\n        this.NUCLEOTIDES = [[1, -1, -1, -1],\n            [-1, 1, -1, -1],\n            [-1, -1, 1, -1],\n            [-1, -1, -1, 1]];\n        this.PAM30 = [[6, -7, -4, -3, -6, -4, -2, -2, -7, -5, -6, -7, -5, -8, -2, 0, -1, -13, -8, -2, -3, -3, -3, -17],\n            [-7, 8, -6, -10, -8, -2, -9, -9, -2, -5, -8, 0, -4, -9, -4, -3, -6, -2, -10, -8, -7, -4, -6, -17],\n            [-4, -6, 8, 2, -11, -3, -2, -3, 0, -5, -7, -1, -9, -9, -6, 0, -2, -8, -4, -8, 6, -3, -3, -17],\n            [-3, -10, 2, 8, -14, -2, 2, -3, -4, -7, -12, -4, -11, -15, -8, -4, -5, -15, -11, -8, 6, 1, -5, -17],\n            [-6, -8, -11, -14, 10, -14, -14, -9, -7, -6, -15, -14, -13, -13, -8, -3, -8, -15, -4, -6, -12, -14, -9, -17],\n            [-4, -2, -3, -2, -14, 8, 1, -7, 1, -8, -5, -3, -4, -13, -3, -5, -5, -13, -12, -7, -3, 6, -5, -17],\n            [-2, -9, -2, 2, -14, 1, 8, -4, -5, -5, -9, -4, -7, -14, -5, -4, -6, -17, -8, -6, 1, 6, -5, -17],\n            [-2, -9, -3, -3, -9, -7, -4, 6, -9, -11, -10, -7, -8, -9, -6, -2, -6, -15, -14, -5, -3, -5, -5, -17],\n            [-7, -2, 0, -4, -7, 1, -5, -9, 9, -9, -6, -6, -10, -6, -4, -6, -7, -7, -3, -6, -1, -1, -5, -17],\n            [-5, -5, -5, -7, -6, -8, -5, -11, -9, 8, -1, -6, -1, -2, -8, -7, -2, -14, -6, 2, -6, -6, -5, -17],\n            [-6, -8, -7, -12, -15, -5, -9, -10, -6, -1, 7, -8, 1, -3, -7, -8, -7, -6, -7, -2, -9, -7, -6, -17],\n            [-7, 0, -1, -4, -14, -3, -4, -7, -6, -6, -8, 7, -2, -14, -6, -4, -3, -12, -9, -9, -2, -4, -5, -17],\n            [-5, -4, -9, -11, -13, -4, -7, -8, -10, -1, 1, -2, 11, -4, -8, -5, -4, -13, -11, -1, -10, -5, -5, -17],\n            [-8, -9, -9, -15, -13, -13, -14, -9, -6, -2, -3, -14, -4, 9, -10, -6, -9, -4, 2, -8, -10, -13, -8, -17],\n            [-2, -4, -6, -8, -8, -3, -5, -6, -4, -8, -7, -6, -8, -10, 8, -2, -4, -14, -13, -6, -7, -4, -5, -17],\n            [0, -3, 0, -4, -3, -5, -4, -2, -6, -7, -8, -4, -5, -6, -2, 6, 0, -5, -7, -6, -1, -5, -3, -17],\n            [-1, -6, -2, -5, -8, -5, -6, -6, -7, -2, -7, -3, -4, -9, -4, 0, 7, -13, -6, -3, -3, -6, -4, -17],\n            [-13, -2, -8, -15, -15, -13, -17, -15, -7, -14, -6, -12, -13, -4, -14, -5, -13, 13, -5, -15, -10, -14, -11, -17],\n            [-8, -10, -4, -11, -4, -12, -8, -14, -3, -6, -7, -9, -11, 2, -13, -7, -6, -5, 10, -7, -6, -9, -7, -17],\n            [-2, -8, -8, -8, -6, -7, -6, -5, -6, 2, -2, -9, -1, -8, -6, -6, -3, -15, -7, 7, -8, -6, -5, -17],\n            [-3, -7, 6, 6, -12, -3, 1, -3, -1, -6, -9, -2, -10, -10, -7, -1, -3, -10, -6, -8, 6, 0, -5, -17],\n            [-3, -4, -3, 1, -14, 6, 6, -5, -1, -6, -7, -4, -5, -13, -4, -5, -6, -14, -9, -6, 0, 6, -5, -17],\n            [-3, -6, -3, -5, -9, -5, -5, -5, -5, -5, -6, -5, -5, -8, -5, -3, -4, -11, -7, -5, -5, -5, -5, -17],\n            [-17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, 1]];\n        this.PAM70 = [[5, -4, -2, -1, -4, -2, -1, 0, -4, -2, -4, -4, -3, -6, 0, 1, 1, -9, -5, -1, -1, -1, -2, -11],\n            [-4, 8, -3, -6, -5, 0, -5, -6, 0, -3, -6, 2, -2, -7, -2, -1, -4, 0, -7, -5, -4, -2, -3, -11],\n            [-2, -3, 6, 3, -7, -1, 0, -1, 1, -3, -5, 0, -5, -6, -3, 1, 0, -6, -3, -5, 5, -1, -2, -11],\n            [-1, -6, 3, 6, -9, 0, 3, -1, -1, -5, -8, -2, -7, -10, -4, -1, -2, -10, -7, -5, 5, 2, -3, -11],\n            [-4, -5, -7, -9, 9, -9, -9, -6, -5, -4, -10, -9, -9, -8, -5, -1, -5, -11, -2, -4, -8, -9, -6, -11],\n            [-2, 0, -1, 0, -9, 7, 2, -4, 2, -5, -3, -1, -2, -9, -1, -3, -3, -8, -8, -4, -1, 5, -2, -11],\n            [-1, -5, 0, 3, -9, 2, 6, -2, -2, -4, -6, -2, -4, -9, -3, -2, -3, -11, -6, -4, 2, 5, -3, -11],\n            [0, -6, -1, -1, -6, -4, -2, 6, -6, -6, -7, -5, -6, -7, -3, 0, -3, -10, -9, -3, -1, -3, -3, -11],\n            [-4, 0, 1, -1, -5, 2, -2, -6, 8, -6, -4, -3, -6, -4, -2, -3, -4, -5, -1, -4, 0, 1, -3, -11],\n            [-2, -3, -3, -5, -4, -5, -4, -6, -6, 7, 1, -4, 1, 0, -5, -4, -1, -9, -4, 3, -4, -4, -3, -11],\n            [-4, -6, -5, -8, -10, -3, -6, -7, -4, 1, 6, -5, 2, -1, -5, -6, -4, -4, -4, 0, -6, -4, -4, -11],\n            [-4, 2, 0, -2, -9, -1, -2, -5, -3, -4, -5, 6, 0, -9, -4, -2, -1, -7, -7, -6, -1, -2, -3, -11],\n            [-3, -2, -5, -7, -9, -2, -4, -6, -6, 1, 2, 0, 10, -2, -5, -3, -2, -8, -7, 0, -6, -3, -3, -11],\n            [-6, -7, -6, -10, -8, -9, -9, -7, -4, 0, -1, -9, -2, 8, -7, -4, -6, -2, 4, -5, -7, -9, -5, -11],\n            [0, -2, -3, -4, -5, -1, -3, -3, -2, -5, -5, -4, -5, -7, 7, 0, -2, -9, -9, -3, -4, -2, -3, -11],\n            [1, -1, 1, -1, -1, -3, -2, 0, -3, -4, -6, -2, -3, -4, 0, 5, 2, -3, -5, -3, 0, -2, -1, -11],\n            [1, -4, 0, -2, -5, -3, -3, -3, -4, -1, -4, -1, -2, -6, -2, 2, 6, -8, -4, -1, -1, -3, -2, -11],\n            [-9, 0, -6, -10, -11, -8, -11, -10, -5, -9, -4, -7, -8, -2, -9, -3, -8, 13, -3, -10, -7, -10, -7, -11],\n            [-5, -7, -3, -7, -2, -8, -6, -9, -1, -4, -4, -7, -7, 4, -9, -5, -4, -3, 9, -5, -4, -7, -5, -11],\n            [-1, -5, -5, -5, -4, -4, -4, -3, -4, 3, 0, -6, 0, -5, -3, -3, -1, -10, -5, 6, -5, -4, -2, -11],\n            [-1, -4, 5, 5, -8, -1, 2, -1, 0, -4, -6, -1, -6, -7, -4, 0, -1, -7, -4, -5, 5, 1, -2, -11],\n            [-1, -2, -1, 2, -9, 5, 5, -3, 1, -4, -4, -2, -3, -9, -2, -2, -3, -10, -7, -4, 1, 5, -3, -11],\n            [-2, -3, -2, -3, -6, -2, -3, -3, -3, -3, -4, -3, -3, -5, -3, -1, -2, -7, -5, -2, -2, -3, -3, -11],\n            [-11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, 1]];\n        this.PAM250 = [[2, -2, 0, 0, -2, 0, 0, 1, -1, -1, -2, -1, -1, -3, 1, 1, 1, -6, -3, 0, 0, 0, 0, -8],\n            [-2, 6, 0, -1, -4, 1, -1, -3, 2, -2, -3, 3, 0, -4, 0, 0, -1, 2, -4, -2, -1, 0, -1, -8],\n            [0, 0, 2, 2, -4, 1, 1, 0, 2, -2, -3, 1, -2, -3, 0, 1, 0, -4, -2, -2, 2, 1, 0, -8],\n            [0, -1, 2, 4, -5, 2, 3, 1, 1, -2, -4, 0, -3, -6, -1, 0, 0, -7, -4, -2, 3, 3, -1, -8],\n            [-2, -4, -4, -5, 12, -5, -5, -3, -3, -2, -6, -5, -5, -4, -3, 0, -2, -8, 0, -2, -4, -5, -3, -8],\n            [0, 1, 1, 2, -5, 4, 2, -1, 3, -2, -2, 1, -1, -5, 0, -1, -1, -5, -4, -2, 1, 3, -1, -8],\n            [0, -1, 1, 3, -5, 2, 4, 0, 1, -2, -3, 0, -2, -5, -1, 0, 0, -7, -4, -2, 3, 3, -1, -8],\n            [1, -3, 0, 1, -3, -1, 0, 5, -2, -3, -4, -2, -3, -5, 0, 1, 0, -7, -5, -1, 0, 0, -1, -8],\n            [-1, 2, 2, 1, -3, 3, 1, -2, 6, -2, -2, 0, -2, -2, 0, -1, -1, -3, 0, -2, 1, 2, -1, -8],\n            [-1, -2, -2, -2, -2, -2, -2, -3, -2, 5, 2, -2, 2, 1, -2, -1, 0, -5, -1, 4, -2, -2, -1, -8],\n            [-2, -3, -3, -4, -6, -2, -3, -4, -2, 2, 6, -3, 4, 2, -3, -3, -2, -2, -1, 2, -3, -3, -1, -8],\n            [-1, 3, 1, 0, -5, 1, 0, -2, 0, -2, -3, 5, 0, -5, -1, 0, 0, -3, -4, -2, 1, 0, -1, -8],\n            [-1, 0, -2, -3, -5, -1, -2, -3, -2, 2, 4, 0, 6, 0, -2, -2, -1, -4, -2, 2, -2, -2, -1, -8],\n            [-3, -4, -3, -6, -4, -5, -5, -5, -2, 1, 2, -5, 0, 9, -5, -3, -3, 0, 7, -1, -4, -5, -2, -8],\n            [1, 0, 0, -1, -3, 0, -1, 0, 0, -2, -3, -1, -2, -5, 6, 1, 0, -6, -5, -1, -1, 0, -1, -8],\n            [1, 0, 1, 0, 0, -1, 0, 1, -1, -1, -3, 0, -2, -3, 1, 2, 1, -2, -3, -1, 0, 0, 0, -8],\n            [1, -1, 0, 0, -2, -1, 0, 0, -1, 0, -2, 0, -1, -3, 0, 1, 3, -5, -3, 0, 0, -1, 0, -8],\n            [-6, 2, -4, -7, -8, -5, -7, -7, -3, -5, -2, -3, -4, 0, -6, -2, -5, 17, 0, -6, -5, -6, -4, -8],\n            [-3, -4, -2, -4, 0, -4, -4, -5, 0, -1, -1, -4, -2, 7, -5, -3, -3, 0, 10, -2, -3, -4, -2, -8],\n            [0, -2, -2, -2, -2, -2, -2, -1, -2, 4, 2, -2, 2, -1, -1, -1, 0, -6, -2, 4, -2, -2, -1, -8],\n            [0, -1, 2, 3, -4, 1, 3, 0, 1, -2, -3, 1, -2, -4, -1, 0, 0, -5, -3, -2, 3, 2, -1, -8],\n            [0, 0, 1, 3, -5, 3, 3, 0, 2, -2, -3, 0, -2, -5, 0, 0, -1, -6, -4, -2, 2, 3, -1, -8],\n            [0, -1, 0, -1, -3, -1, -1, -1, -1, -1, -1, -1, -1, -2, -1, 0, 0, -4, -2, -1, -1, -1, -1, -8],\n            [-8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, 1]];\n        this.SCHNEIDER = [[11.6, -2.7, 9.7, -1.7, -2.7, -6.4, -3.9, -5.6, 5.1, -5.0, 3.6, -4.2, -6.3, -13.0, -7.1, -11.5, 0.4, -6.0, -1.9, -5.3, -8.5, -11.2, -8.9, -10.8, 2.1, 0.0, 1.4, 0.2, -10.2, -13.5, -13.0, -12.5, -2.6, -8.5, -5.0, -8.1, -6.3, -9.9, -7.5, -9.0, -7.1, -10.2, -8.2, -9.2, -8.2, -12.5, -11.1, -11.4, -50.0, -14.8, -50.0, -13.8, -7.3, -10.1, -8.4, -9.1, -50.0, -13.0, -13.5, -12.4, -10.7, -18.1, -11.8, -17.2,],\n            [-2.7, 13.0, -3.3, 10.9, -3.5, -0.4, -3.3, -1.8, -5.4, 4.6, -5.5, 3.0, -10.2, -7.9, -9.9, -9.6, -5.0, 0.5, -5.5, -1.0, -10.3, -8.1, -9.4, -9.6, -8.1, -5.0, -7.3, -6.3, -13.4, -11.3, -14.4, -12.9, -6.3, 0.8, -6.4, -1.1, -7.4, -5.0, -6.2, -6.5, -5.6, -1.6, -4.7, -3.0, -10.8, -8.7, -11.9, -10.0, -50.0, -6.2, -50.0, -7.5, -6.3, -4.3, -6.2, -5.4, -50.0, -7.0, -16.3, -8.2, -13.2, -12.3, -13.1, -13.3,],\n            [9.7, -3.3, 11.6, -2.8, -4.5, -6.7, -3.1, -6.9, 3.3, -5.5, 4.8, -5.1, -8.9, -13.2, -5.7, -12.6, -1.5, -6.1, -0.6, -6.1, -10.0, -11.8, -8.6, -11.9, 1.2, 0.5, 2.2, 0.1, -11.8, -14.0, -11.9, -13.4, -4.9, -9.1, -3.4, -8.9, -8.0, -10.0, -7.2, -10.0, -9.1, -10.2, -7.1, -9.9, -10.1, -13.0, -10.6, -12.7, -50.0, -14.9, -50.0, -14.4, -8.9, -10.8, -8.9, -10.4, -50.0, -13.1, -11.8, -13.0, -12.4, -19.4, -11.5, -17.8,],\n            [-1.7, 10.9, -2.8, 12.9, -2.7, -2.2, -2.8, 0.2, -4.8, 2.9, -5.2, 5.2, -9.1, -9.5, -9.0, -7.0, -4.0, -1.0, -5.0, 1.0, -9.0, -8.8, -8.9, -7.7, -7.0, -6.8, -7.1, -4.3, -12.4, -12.9, -13.9, -10.0, -4.9, -0.7, -5.6, 1.4, -6.0, -6.5, -6.2, -4.8, -5.1, -3.2, -5.0, -1.2, -9.8, -9.9, -11.3, -8.2, -50.0, -7.6, -50.0, -5.1, -5.5, -5.8, -5.9, -4.2, -50.0, -8.3, -15.5, -5.9, -11.6, -14.2, -12.0, -11.6,],\n            [-2.7, -3.5, -4.5, -2.7, 11.7, 9.0, 10.6, 9.6, -3.2, -0.7, -5.1, -0.4, 0.7, -4.6, -0.4, -3.6, -4.8, -8.9, -6.5, -7.8, -1.4, -4.1, -2.9, -3.3, -8.0, -9.1, -8.0, -8.2, -6.0, -9.3, -7.7, -8.5, -6.4, -10.1, -7.5, -8.6, 2.8, -0.7, 0.9, 0.1, -5.9, -7.3, -6.5, -6.5, -0.4, -3.9, -2.6, -3.1, -50.0, -14.6, -50.0, -12.6, 2.9, 0.1, 1.8, 1.0, -50.0, -9.2, -13.6, -7.7, -4.8, -12.2, -6.2, -11.0,],\n            [-6.4, -0.4, -6.7, -2.2, 9.0, 12.3, 9.8, 9.6, -7.1, 2.4, -7.2, 0.3, -2.9, -1.2, -3.0, -3.3, -7.6, -6.7, -8.1, -7.9, -4.9, -1.9, -4.5, -3.8, -9.5, -7.8, -9.3, -8.9, -9.4, -7.1, -9.0, -9.0, -9.4, -7.5, -9.3, -9.2, -0.7, 2.6, 0.2, -0.1, -8.0, -4.6, -7.7, -6.1, -3.6, -1.0, -4.1, -3.1, -50.0, -11.6, -50.0, -11.6, 0.2, 2.3, 0.4, 0.5, -50.0, -6.5, -15.3, -7.8, -8.4, -9.5, -8.3, -11.1,],\n            [-3.9, -3.3, -3.1, -2.8, 10.6, 9.8, 12.2, 9.8, -4.9, 0.1, -3.6, -0.2, -1.7, -4.1, 0.9, -3.5, -5.9, -8.9, -5.6, -8.5, -3.9, -3.9, -2.2, -4.5, -7.3, -7.5, -6.4, -8.5, -7.1, -8.2, -6.7, -7.8, -7.9, -9.1, -7.1, -9.4, 0.9, -0.2, 2.5, -0.3, -6.9, -6.5, -5.5, -6.8, -1.7, -3.4, -1.7, -2.8, -50.0, -13.0, -50.0, -11.5, 1.5, 0.7, 2.7, 0.6, -50.0, -8.9, -12.1, -7.4, -6.1, -11.5, -5.4, -10.4,],\n            [-5.6, -1.8, -6.9, 0.2, 9.6, 9.6, 9.8, 11.6, -6.6, 0.9, -7.2, 2.5, -2.3, -3.4, -2.3, -0.6, -6.5, -8.2, -8.0, -6.3, -3.7, -3.5, -3.9, -1.6, -9.9, -8.9, -9.6, -7.0, -8.6, -8.9, -8.8, -6.8, -8.6, -8.2, -8.8, -6.8, 0.3, 0.1, 0.2, 2.4, -7.4, -6.2, -6.9, -4.4, -2.9, -2.9, -3.5, -0.9, -50.0, -12.2, -50.0, -10.1, 1.2, 0.6, 1.2, 2.4, -50.0, -7.6, -16.1, -5.8, -7.2, -10.8, -7.4, -9.0,],\n            [5.1, -5.4, 3.3, -4.8, -3.2, -7.1, -4.9, -6.6, 13.3, -2.8, 11.2, -1.9, -5.7, -12.5, -7.2, -11.6, -0.7, -4.8, -3.1, -4.3, -10.0, -11.6, -8.8, -11.7, 10.5, 7.7, 9.1, 8.5, -9.8, -12.7, -11.7, -11.8, -6.3, -11.4, -8.8, -11.0, -7.5, -10.3, -8.3, -9.9, -1.9, -6.7, -4.1, -6.4, -7.9, -12.5, -11.4, -11.7, -50.0, -14.0, -50.0, -13.2, -8.5, -10.4, -9.1, -9.6, -50.0, -9.8, -7.7, -8.8, -10.3, -17.9, -11.8, -16.0,],\n            [-5.0, 4.6, -5.5, 2.9, -0.7, 2.4, 0.1, 0.9, -2.8, 12.8, -2.6, 11.0, -8.4, -6.3, -8.3, -7.7, -6.1, -2.9, -6.5, -4.2, -8.5, -5.9, -7.5, -7.7, -5.5, -2.4, -5.5, -4.1, -12.9, -10.3, -12.7, -11.3, -7.4, -3.1, -7.4, -4.9, -4.4, -1.8, -3.5, -3.5, -1.6, 3.0, -0.8, 0.9, -8.4, -6.3, -9.6, -7.3, -50.0, -9.0, -50.0, -9.7, -2.3, -0.2, -1.2, -1.5, -50.0, -0.7, -13.3, -2.0, -11.8, -11.9, -12.3, -12.7,],\n            [3.6, -5.5, 4.8, -5.2, -5.1, -7.2, -3.6, -7.2, 11.2, -2.6, 13.4, -2.1, -7.7, -12.5, -5.3, -11.9, -2.2, -4.4, -1.9, -4.8, -10.2, -11.5, -8.1, -12.0, 9.3, 8.2, 10.0, 8.0, -10.8, -11.9, -10.7, -12.4, -8.2, -11.8, -6.9, -11.5, -8.4, -9.9, -7.1, -10.6, -4.7, -6.8, -1.2, -7.0, -9.8, -11.6, -9.6, -12.1, -50.0, -14.1, -50.0, -12.7, -10.0, -10.8, -8.8, -10.5, -50.0, -9.2, -4.2, -9.3, -11.0, -18.2, -11.1, -16.1,],\n            [-4.2, 3.0, -5.1, 5.2, -0.4, 0.3, -0.2, 2.5, -1.9, 11.0, -2.1, 13.2, -7.6, -8.5, -8.0, -5.3, -5.6, -4.5, -6.8, -2.4, -7.8, -7.9, -7.9, -6.2, -5.7, -5.1, -5.4, -2.1, -12.4, -12.3, -13.1, -10.1, -6.6, -4.4, -6.9, -2.6, -3.6, -3.8, -3.7, -2.0, -1.4, 0.8, -0.9, 3.0, -8.2, -8.1, -9.2, -5.8, -50.0, -10.1, -50.0, -7.4, -1.7, -2.0, -1.5, -0.5, -50.0, -2.1, -12.6, -0.4, -11.6, -13.9, -11.4, -11.1,],\n            [-6.3, -10.2, -8.9, -9.1, 0.7, -2.9, -1.7, -2.3, -5.7, -8.4, -7.7, -7.6, 13.2, 9.6, 3.5, 9.7, -8.7, -12.4, -10.7, -10.7, -7.8, -10.2, -8.8, -9.6, -9.5, -11.9, -10.9, -10.5, 2.3, -0.4, -0.2, -0.3, -9.7, -15.5, -11.4, -13.7, -3.0, -6.4, -4.1, -5.4, -9.6, -11.9, -10.5, -11.8, 6.2, 3.3, 3.7, 3.6, -50.0, -13.6, -50.0, -11.9, -5.6, -8.8, -7.2, -8.9, -50.0, -12.4, -14.1, -11.6, 2.8, -6.4, 0.5, -5.2,],\n            [-13.0, -7.9, -13.2, -9.5, -4.6, -1.2, -4.1, -3.4, -12.5, -6.3, -12.5, -8.5, 9.6, 12.7, 0.2, 10.5, -12.4, -11.6, -13.2, -12.0, -12.4, -10.1, -11.6, -12.0, -15.1, -13.1, -14.3, -13.4, -1.4, 1.4, -1.6, -0.5, -14.9, -14.3, -15.2, -16.8, -7.7, -4.7, -6.3, -6.6, -14.6, -10.8, -12.8, -13.0, 2.3, 6.0, 2.5, 3.6, -50.0, -11.2, -50.0, -11.9, -10.5, -9.2, -10.5, -10.5, -50.0, -10.3, -16.2, -11.3, -1.7, -3.7, -2.5, -5.4,],\n            [-7.1, -9.9, -5.7, -9.0, -0.4, -3.0, 0.9, -2.3, -7.2, -8.3, -5.3, -8.0, 3.5, 0.2, 14.3, 1.0, -7.3, -11.2, -6.9, -9.7, -8.5, -10.2, -7.7, -9.8, -10.0, -10.4, -8.4, -10.2, 1.0, -0.6, 1.7, -0.5, -11.1, -14.8, -9.7, -14.5, -3.7, -5.8, -3.1, -5.6, -10.4, -11.5, -8.2, -11.1, 0.6, -1.6, 1.6, -1.1, -50.0, -12.4, -50.0, -11.5, -5.5, -8.0, -4.9, -7.2, -50.0, -12.4, -10.3, -11.6, 0.7, -6.8, 2.4, -6.0,],\n            [-11.5, -9.6, -12.6, -7.0, -3.6, -3.3, -3.5, -0.6, -11.6, -7.7, -11.9, -5.3, 9.7, 10.5, 1.0, 12.6, -12.0, -12.2, -12.5, -9.7, -11.1, -11.2, -11.9, -9.4, -12.9, -14.3, -14.0, -11.9, -1.0, -0.4, -1.6, 1.2, -13.9, -15.4, -13.7, -12.6, -6.2, -6.7, -6.3, -4.4, -13.0, -12.2, -13.0, -10.4, 2.9, 3.8, 2.5, 5.8, -50.0, -12.5, -50.0, -10.2, -9.6, -9.7, -9.6, -8.3, -50.0, -11.6, -15.4, -9.4, -1.1, -5.4, -1.6, -3.3,],\n            [0.4, -5.0, -1.5, -4.0, -4.8, -7.6, -5.9, -6.5, -0.7, -6.1, -2.2, -5.6, -8.7, -12.4, -7.3, -12.0, 12.8, 2.3, 10.2, 3.0, 0.0, -3.4, -0.7, -3.2, 2.5, -0.8, 0.9, 0.2, -3.0, -7.2, -5.7, -6.1, -0.2, -6.6, -1.8, -6.0, -5.3, -8.2, -5.7, -7.6, -6.7, -9.6, -7.5, -9.3, -7.1, -10.5, -9.6, -9.4, -50.0, -8.1, -50.0, -7.2, -4.3, -6.8, -5.7, -6.3, -50.0, -9.5, -9.6, -8.9, -6.2, -12.8, -6.8, -11.8,],\n            [-6.0, 0.5, -6.1, -1.0, -8.9, -6.7, -8.9, -8.2, -4.8, -2.9, -4.4, -4.5, -12.4, -11.6, -11.2, -12.2, 2.3, 14.6, 1.9, 12.9, -5.5, -2.6, -4.1, -4.7, -1.2, 3.0, -1.1, 1.6, -8.0, -4.7, -8.7, -6.0, -8.2, -4.7, -7.8, -6.7, -10.0, -8.3, -8.9, -9.9, -10.8, -7.1, -10.0, -9.2, -11.6, -10.1, -12.3, -11.9, -50.0, 2.2, -50.0, 0.9, -7.9, -5.9, -7.4, -6.9, -50.0, -4.6, -11.6, -5.6, -9.7, -5.9, -9.9, -7.3,],\n            [-1.9, -5.5, -0.6, -5.0, -6.5, -8.1, -5.6, -8.0, -3.1, -6.5, -1.9, -6.8, -10.7, -13.2, -6.9, -12.5, 10.2, 1.9, 11.9, 2.1, -2.3, -4.1, -0.5, -4.6, -0.1, -0.9, 2.4, -0.8, -5.1, -6.9, -4.8, -6.7, -2.3, -7.1, -0.8, -7.2, -7.2, -8.0, -5.6, -8.5, -9.4, -9.7, -7.2, -9.8, -9.2, -11.1, -9.5, -10.7, -50.0, -8.7, -50.0, -8.6, -6.1, -7.5, -5.5, -7.7, -50.0, -10.4, -7.7, -10.0, -7.3, -13.6, -6.4, -13.3,],\n            [-5.3, -1.0, -6.1, 1.0, -7.8, -7.9, -8.5, -6.3, -4.3, -4.2, -4.8, -2.4, -10.7, -12.0, -9.7, -9.7, 3.0, 12.9, 2.1, 14.7, -4.9, -3.8, -4.0, -2.4, -1.0, 0.8, -1.3, 3.4, -7.6, -5.7, -8.1, -3.5, -6.8, -5.8, -7.2, -4.2, -8.9, -10.2, -8.4, -8.2, -9.9, -8.4, -9.7, -6.4, -11.1, -11.1, -11.8, -10.0, -50.0, 0.4, -50.0, 2.6, -7.2, -7.0, -7.4, -5.1, -50.0, -5.7, -9.8, -3.2, -8.4, -7.4, -8.9, -5.9,],\n            [-8.5, -10.3, -10.0, -9.0, -1.4, -4.9, -3.9, -3.7, -10.0, -8.5, -10.2, -7.8, -7.8, -12.4, -8.5, -11.1, 0.0, -5.5, -2.3, -4.9, 12.6, 10.0, 11.1, 10.5, -6.0, -8.3, -6.9, -7.5, -2.2, -7.4, -5.3, -5.9, -8.9, -12.5, -9.4, -11.6, -0.7, -3.9, -2.3, -3.0, -9.1, -9.8, -9.2, -9.5, -5.6, -9.0, -8.0, -8.5, -50.0, -15.3, -50.0, -14.1, 2.3, -1.4, 0.3, -0.5, -50.0, -13.6, -14.3, -11.5, -5.3, -13.7, -6.1, -12.7,],\n            [-11.2, -8.1, -11.8, -8.8, -4.1, -1.9, -3.9, -3.5, -11.6, -5.9, -11.5, -7.9, -10.2, -10.1, -10.2, -11.2, -3.4, -2.6, -4.1, -3.8, 10.0, 13.1, 10.7, 10.6, -8.1, -5.2, -8.0, -6.7, -6.1, -3.7, -7.5, -5.6, -11.1, -10.3, -10.6, -11.9, -3.2, -1.0, -2.2, -2.8, -10.5, -8.1, -9.4, -9.5, -8.3, -6.9, -9.4, -8.8, -50.0, -11.3, -50.0, -12.8, -0.6, 2.2, -0.3, 0.1, -50.0, -10.1, -17.5, -11.1, -8.6, -9.9, -8.6, -11.9,],\n            [-8.9, -9.4, -8.6, -8.9, -2.9, -4.5, -2.2, -3.9, -8.8, -7.5, -8.1, -7.9, -8.8, -11.6, -7.7, -11.9, -0.7, -4.1, -0.5, -4.0, 11.1, 10.7, 13.2, 10.4, -5.7, -5.8, -3.5, -6.1, -3.6, -6.5, -3.3, -5.5, -9.4, -11.0, -8.7, -11.2, -1.8, -2.7, 0.7, -3.0, -9.1, -8.2, -7.3, -9.4, -7.2, -8.2, -7.2, -8.2, -50.0, -13.2, -50.0, -13.3, 0.5, -0.9, 1.8, -0.6, -50.0, -11.4, -11.1, -10.5, -6.1, -12.9, -5.0, -11.9,],\n            [-10.8, -9.6, -11.9, -7.7, -3.3, -3.8, -4.5, -1.6, -11.7, -7.7, -12.0, -6.2, -9.6, -12.0, -9.8, -9.4, -3.2, -4.7, -4.6, -2.4, 10.5, 10.6, 10.4, 12.6, -8.2, -7.9, -9.0, -4.7, -6.1, -6.0, -7.5, -2.9, -10.9, -12.2, -11.2, -10.9, -2.4, -2.9, -2.7, -0.9, -10.4, -9.7, -10.2, -8.0, -8.4, -9.1, -9.3, -6.5, -50.0, -13.3, -50.0, -11.0, -0.0, -0.4, -0.4, 2.3, -50.0, -10.9, -17.2, -8.3, -8.3, -12.7, -7.8, -9.3,],\n            [2.1, -8.1, 1.2, -7.0, -8.0, -9.5, -7.3, -9.9, 10.5, -5.5, 9.3, -5.7, -9.5, -15.1, -10.0, -12.9, 2.5, -1.2, -0.1, -1.0, -6.0, -8.1, -5.7, -8.2, 13.8, 11.3, 11.8, 12.1, -6.0, -9.1, -9.0, -9.5, -8.9, -13.4, -10.0, -13.1, -9.9, -11.7, -9.8, -12.8, -5.9, -9.4, -6.5, -8.1, -10.7, -14.0, -13.3, -13.9, -50.0, -11.9, -50.0, -9.7, -8.9, -10.8, -9.0, -11.1, -50.0, -6.5, -5.3, -6.3, -10.5, -16.2, -10.4, -15.2,],\n            [0.0, -5.0, 0.5, -6.8, -9.1, -7.8, -7.5, -8.9, 7.7, -2.4, 8.2, -5.1, -11.9, -13.1, -10.4, -14.3, -0.8, 3.0, -0.9, 0.8, -8.3, -5.2, -5.8, -7.9, 11.3, 15.0, 11.2, 12.8, -9.7, -5.3, -9.2, -7.8, -11.7, -10.6, -10.1, -12.2, -11.7, -8.8, -8.9, -11.8, -9.3, -5.0, -8.0, -8.2, -13.6, -11.4, -12.8, -13.0, -50.0, -7.2, -50.0, -8.4, -11.0, -7.7, -8.7, -10.2, -50.0, -1.5, -7.1, -4.9, -11.2, -11.6, -11.6, -15.0,],\n            [1.4, -7.3, 2.2, -7.1, -8.0, -9.3, -6.4, -9.6, 9.1, -5.5, 10.0, -5.4, -10.9, -14.3, -8.4, -14.0, 0.9, -1.1, 2.4, -1.3, -6.9, -8.0, -3.5, -9.0, 11.8, 11.2, 13.4, 11.4, -7.3, -8.8, -6.5, -8.5, -10.0, -12.5, -7.8, -12.5, -9.6, -11.0, -7.1, -10.7, -7.9, -8.0, -4.3, -9.4, -11.4, -12.3, -10.9, -12.9, -50.0, -11.8, -50.0, -11.2, -9.4, -9.8, -7.3, -10.5, -50.0, -6.2, -2.2, -6.7, -9.8, -15.2, -8.5, -14.9,],\n            [0.2, -6.3, 0.1, -4.3, -8.2, -8.9, -8.5, -7.0, 8.5, -4.1, 8.0, -2.1, -10.5, -13.4, -10.2, -11.9, 0.2, 1.6, -0.8, 3.4, -7.5, -6.7, -6.1, -4.7, 12.1, 12.8, 11.4, 14.7, -8.7, -7.4, -9.4, -5.3, -9.9, -11.8, -10.5, -10.3, -9.9, -10.1, -9.3, -9.3, -8.1, -7.2, -8.0, -5.2, -12.0, -12.1, -12.6, -10.9, -50.0, -7.9, -50.0, -5.3, -9.3, -8.8, -8.9, -7.0, -50.0, -3.6, -7.1, -1.2, -10.2, -12.7, -9.9, -11.0,],\n            [-10.2, -13.4, -11.8, -12.4, -6.0, -9.4, -7.1, -8.6, -9.8, -12.9, -10.8, -12.4, 2.3, -1.4, 1.0, -1.0, -3.0, -8.0, -5.1, -7.6, -2.2, -6.1, -3.6, -6.1, -6.0, -9.7, -7.3, -8.7, 11.2, 7.9, 8.9, 8.1, -12.1, -17.4, -13.2, -16.8, -6.3, -9.5, -7.5, -8.8, -12.7, -15.2, -13.1, -14.2, 0.0, -3.1, -1.8, -3.1, -50.0, -10.7, -50.0, -9.3, -4.6, -8.9, -5.9, -8.3, -50.0, -12.8, -9.6, -10.6, 9.5, -3.2, 8.2, -2.8,],\n            [-13.5, -11.3, -14.0, -12.9, -9.3, -7.1, -8.2, -8.9, -12.7, -10.3, -11.9, -12.3, -0.4, 1.4, -0.6, -0.4, -7.2, -4.7, -6.9, -5.7, -7.4, -3.7, -6.5, -6.0, -9.1, -5.3, -8.8, -7.4, 7.9, 11.9, 7.8, 9.3, -15.2, -15.1, -14.6, -17.2, -9.7, -7.3, -8.2, -9.2, -15.0, -12.8, -14.1, -14.9, -2.8, 0.2, -2.9, -2.3, -50.0, -7.4, -50.0, -8.7, -9.4, -7.3, -9.4, -9.4, -50.0, -8.0, -11.9, -9.7, 6.5, 0.5, 6.4, -1.8,],\n            [-13.0, -14.4, -11.9, -13.9, -7.7, -9.0, -6.7, -8.8, -11.7, -12.7, -10.7, -13.1, -0.2, -1.6, 1.7, -1.6, -5.7, -8.7, -4.8, -8.1, -5.3, -7.5, -3.3, -7.5, -9.0, -9.2, -6.5, -9.4, 8.9, 7.8, 10.1, 7.8, -14.3, -17.7, -13.0, -17.8, -8.0, -9.4, -6.2, -9.2, -15.1, -14.9, -12.2, -14.9, -1.8, -3.2, -0.8, -3.2, -50.0, -10.8, -50.0, -9.9, -7.3, -9.6, -6.4, -9.4, -50.0, -12.3, -8.7, -11.3, 7.6, -3.3, 8.7, -2.9,],\n            [-12.5, -12.9, -13.4, -10.0, -8.5, -9.0, -7.8, -6.8, -11.8, -11.3, -12.4, -10.1, -0.3, -0.5, -0.5, 1.2, -6.1, -6.0, -6.7, -3.5, -5.9, -5.6, -5.5, -2.9, -9.5, -7.8, -8.5, -5.3, 8.1, 9.3, 7.8, 11.8, -14.4, -15.7, -14.5, -14.6, -8.4, -8.7, -7.9, -7.1, -14.3, -13.8, -13.8, -12.3, -2.4, -2.2, -2.9, -0.2, -50.0, -8.9, -50.0, -7.1, -8.1, -9.1, -8.7, -6.6, -50.0, -9.7, -11.4, -8.0, 6.8, -1.8, 6.9, 0.4,],\n            [-2.6, -6.3, -4.9, -4.9, -6.4, -9.4, -7.9, -8.6, -6.3, -7.4, -8.2, -6.6, -9.7, -14.9, -11.1, -13.9, -0.2, -8.2, -2.3, -6.8, -8.9, -11.1, -9.4, -10.9, -8.9, -11.7, -10.0, -9.9, -12.1, -15.2, -14.3, -14.4, 11.1, 2.9, 9.2, 3.5, -3.0, -7.0, -4.3, -6.2, -2.2, -6.6, -3.7, -5.7, -6.0, -10.4, -8.6, -9.5, -50.0, -15.5, -50.0, -13.9, -7.8, -10.5, -8.7, -9.6, -50.0, -16.5, -17.1, -14.8, -12.1, -18.9, -12.4, -17.4,],\n            [-8.5, 0.8, -9.1, -0.7, -10.1, -7.5, -9.1, -8.2, -11.4, -3.1, -11.8, -4.4, -15.5, -14.3, -14.8, -15.4, -6.6, -4.7, -7.1, -5.8, -12.5, -10.3, -11.0, -12.2, -13.4, -10.6, -12.5, -11.8, -17.4, -15.1, -17.7, -15.7, 2.9, 12.5, 3.1, 10.3, -7.7, -5.2, -6.2, -7.3, -5.4, -1.7, -5.4, -3.7, -11.6, -9.1, -13.0, -10.8, -50.0, -10.3, -50.0, -11.2, -9.8, -8.3, -9.1, -9.7, -50.0, -12.7, -20.6, -13.1, -16.0, -16.6, -17.0, -18.4,],\n            [-5.0, -6.4, -3.4, -5.6, -7.5, -9.3, -7.1, -8.8, -8.8, -7.4, -6.9, -6.9, -11.4, -15.2, -9.7, -13.7, -1.8, -7.8, -0.8, -7.2, -9.4, -10.6, -8.7, -11.2, -10.0, -10.1, -7.8, -10.5, -13.2, -14.6, -13.0, -14.5, 9.2, 3.1, 10.8, 3.0, -4.5, -6.7, -2.7, -6.7, -5.0, -6.8, -2.5, -6.7, -7.5, -10.4, -7.4, -10.2, -50.0, -15.5, -50.0, -14.2, -8.9, -10.2, -8.5, -10.0, -50.0, -15.6, -15.3, -15.5, -13.0, -18.8, -12.6, -18.3,],\n            [-8.1, -1.1, -8.9, 1.4, -8.6, -9.2, -9.4, -6.8, -11.0, -4.9, -11.5, -2.6, -13.7, -16.8, -14.5, -12.6, -6.0, -6.7, -7.2, -4.2, -11.6, -11.9, -11.2, -10.9, -13.1, -12.2, -12.5, -10.3, -16.8, -17.2, -17.8, -14.6, 3.5, 10.3, 3.0, 12.4, -6.9, -7.5, -6.8, -4.9, -5.4, -4.2, -5.4, -1.3, -10.8, -11.5, -12.4, -8.2, -50.0, -12.5, -50.0, -8.9, -9.2, -10.0, -9.4, -7.9, -50.0, -14.3, -19.9, -11.4, -16.2, -19.2, -15.8, -15.9,],\n            [-6.3, -7.4, -8.0, -6.0, 2.8, -0.7, 0.9, 0.3, -7.5, -4.4, -8.4, -3.6, -3.0, -7.7, -3.7, -6.2, -5.3, -10.0, -7.2, -8.9, -0.7, -3.2, -1.8, -2.4, -9.9, -11.7, -9.6, -9.9, -6.3, -9.7, -8.0, -8.4, -3.0, -7.7, -4.5, -6.9, 11.3, 8.2, 9.4, 9.1, -1.0, -3.3, -1.8, -2.4, 1.5, -2.1, -0.4, -1.1, -50.0, -14.5, -50.0, -12.9, 2.8, -0.0, 1.4, 0.7, -50.0, -8.7, -14.1, -7.5, -5.2, -12.5, -6.5, -10.7,],\n            [-9.9, -5.0, -10.0, -6.5, -0.7, 2.6, -0.2, 0.1, -10.3, -1.8, -9.9, -3.8, -6.4, -4.7, -5.8, -6.7, -8.2, -8.3, -8.0, -10.2, -3.9, -1.0, -2.7, -2.9, -11.7, -8.8, -11.0, -10.1, -9.5, -7.3, -9.4, -8.7, -7.0, -5.2, -6.7, -7.5, 8.2, 11.6, 8.8, 9.1, -3.7, -0.9, -3.1, -3.0, -2.0, 1.2, -2.3, -1.3, -50.0, -12.1, -50.0, -12.6, 0.1, 2.5, 0.7, 0.6, -50.0, -6.0, -14.8, -7.5, -8.6, -9.4, -8.5, -11.2,],\n            [-7.5, -6.2, -7.2, -6.2, 0.9, 0.2, 2.5, 0.2, -8.3, -3.5, -7.1, -3.7, -4.1, -6.3, -3.1, -6.3, -5.7, -8.9, -5.6, -8.4, -2.3, -2.2, 0.7, -2.7, -9.8, -8.9, -7.1, -9.3, -7.5, -8.2, -6.2, -7.9, -4.3, -6.2, -2.7, -6.8, 9.4, 8.8, 12.1, 8.8, -2.1, -2.0, 0.1, -2.4, 0.0, -1.1, 1.3, -0.8, -50.0, -11.9, -50.0, -12.6, 1.2, 0.7, 3.5, 0.8, -50.0, -7.1, -12.4, -7.1, -7.1, -10.6, -5.0, -10.6,],\n            [-9.0, -6.5, -10.0, -4.8, 0.1, -0.1, -0.3, 2.4, -9.9, -3.5, -10.6, -2.0, -5.4, -6.6, -5.6, -4.4, -7.6, -9.9, -8.5, -8.2, -3.0, -2.8, -3.0, -0.9, -12.8, -11.8, -10.7, -9.3, -8.8, -9.2, -9.2, -7.1, -6.2, -7.3, -6.7, -4.9, 9.1, 9.1, 8.8, 11.2, -3.1, -3.0, -3.3, -1.0, -1.1, -1.4, -2.0, 1.2, -50.0, -12.8, -50.0, -10.9, 0.8, 0.5, 0.7, 2.3, -50.0, -7.5, -16.5, -5.9, -7.8, -11.1, -7.8, -9.0,],\n            [-7.1, -5.6, -9.1, -5.1, -5.9, -8.0, -6.9, -7.4, -1.9, -1.6, -4.7, -1.4, -9.6, -14.6, -10.4, -13.0, -6.7, -10.8, -9.4, -9.9, -9.1, -10.5, -9.1, -10.4, -5.9, -9.3, -7.9, -8.1, -12.7, -15.0, -15.1, -14.3, -2.2, -5.4, -5.0, -5.4, -1.0, -3.7, -2.1, -3.1, 12.8, 9.6, 11.1, 10.1, -4.8, -9.4, -8.1, -8.6, -50.0, -18.3, -50.0, -15.4, -5.9, -7.5, -6.6, -7.1, -50.0, -9.5, -11.7, -8.4, -12.5, -17.3, -13.2, -15.4,],\n            [-10.2, -1.6, -10.2, -3.2, -7.3, -4.6, -6.5, -6.2, -6.7, 3.0, -6.8, 0.8, -11.9, -10.8, -11.5, -12.2, -9.6, -7.1, -9.7, -8.4, -9.8, -8.1, -8.2, -9.7, -9.4, -5.0, -8.0, -7.2, -15.2, -12.8, -14.9, -13.8, -6.6, -1.7, -6.8, -4.2, -3.3, -0.9, -2.0, -3.0, 9.6, 12.8, 9.7, 10.5, -8.5, -5.6, -9.3, -7.8, -50.0, -12.7, -50.0, -13.8, -6.8, -5.1, -5.3, -6.5, -50.0, -4.1, -12.8, -5.8, -14.4, -14.1, -13.6, -14.8,],\n            [-8.2, -4.7, -7.1, -5.0, -6.5, -7.7, -5.5, -6.9, -4.1, -0.8, -1.2, -0.9, -10.5, -12.8, -8.2, -13.0, -7.5, -10.0, -7.2, -9.7, -9.2, -9.4, -7.3, -10.2, -6.5, -8.0, -4.3, -8.0, -13.1, -14.1, -12.2, -13.8, -3.7, -5.4, -2.5, -5.4, -1.8, -3.1, 0.1, -3.3, 11.1, 9.7, 12.9, 9.8, -5.5, -8.1, -5.4, -7.8, -50.0, -16.4, -50.0, -14.8, -5.9, -7.1, -4.7, -7.2, -50.0, -8.4, -6.4, -8.1, -12.1, -16.2, -10.3, -15.8,],\n            [-9.2, -3.0, -9.9, -1.2, -6.5, -6.1, -6.8, -4.4, -6.4, 0.9, -7.0, 3.0, -11.8, -13.0, -11.1, -10.4, -9.3, -9.2, -9.8, -6.4, -9.5, -9.5, -9.4, -8.0, -8.1, -8.2, -9.4, -5.2, -14.2, -14.9, -14.9, -12.3, -5.7, -3.7, -6.7, -1.3, -2.4, -3.0, -2.4, -1.0, 10.1, 10.5, 9.8, 13.1, -7.2, -8.0, -8.4, -5.0, -50.0, -14.7, -50.0, -11.4, -6.3, -6.7, -5.8, -5.0, -50.0, -6.1, -13.4, -3.6, -13.0, -16.0, -13.7, -13.3,],\n            [-8.2, -10.8, -10.1, -9.8, -0.4, -3.6, -1.7, -2.9, -7.9, -8.4, -9.8, -8.2, 6.2, 2.3, 0.6, 2.9, -7.1, -11.6, -9.2, -11.1, -5.6, -8.3, -7.2, -8.4, -10.7, -13.6, -11.4, -12.0, 0.0, -2.8, -1.8, -2.4, -6.0, -11.6, -7.5, -10.8, 1.5, -2.0, 0.0, -1.1, -4.8, -8.5, -5.5, -7.2, 11.9, 8.6, 10.0, 9.0, -50.0, -14.2, -50.0, -12.8, -4.1, -7.4, -5.7, -6.5, -50.0, -11.0, -14.3, -9.9, 1.4, -7.9, -0.7, -6.8,],\n            [-12.5, -8.7, -13.0, -9.9, -3.9, -1.0, -3.4, -2.9, -12.5, -6.3, -11.6, -8.1, 3.3, 6.0, -1.6, 3.8, -10.5, -10.1, -11.1, -11.1, -9.0, -6.9, -8.2, -9.1, -14.0, -11.4, -12.3, -12.1, -3.1, 0.2, -3.2, -2.2, -10.4, -9.1, -10.4, -11.5, -2.1, 1.2, -1.1, -1.4, -9.4, -5.6, -8.1, -8.0, 8.6, 12.4, 8.6, 9.5, -50.0, -10.4, -50.0, -11.6, -7.2, -5.7, -7.5, -7.4, -50.0, -8.6, -15.7, -9.7, -3.0, -3.7, -3.2, -5.9,],\n            [-11.1, -11.9, -10.6, -11.3, -2.6, -4.1, -1.7, -3.5, -11.4, -9.6, -9.6, -9.2, 3.7, 2.5, 1.6, 2.5, -9.6, -12.3, -9.5, -11.8, -8.0, -9.4, -7.2, -9.3, -13.3, -12.8, -10.9, -12.6, -1.8, -2.9, -0.8, -2.9, -8.6, -13.0, -7.4, -12.4, -0.4, -2.3, 1.3, -2.0, -8.1, -9.3, -5.4, -8.4, 10.0, 8.6, 11.4, 8.9, -50.0, -13.7, -50.0, -13.2, -6.1, -8.1, -5.5, -7.5, -50.0, -11.3, -12.7, -10.1, -1.6, -7.8, -0.3, -7.2,],\n            [-11.4, -10.0, -12.7, -8.2, -3.1, -3.1, -2.8, -0.9, -11.7, -7.3, -12.1, -5.8, 3.6, 3.6, -1.1, 5.8, -9.4, -11.9, -10.7, -10.0, -8.5, -8.8, -8.2, -6.5, -13.9, -13.0, -12.9, -10.9, -3.1, -2.3, -3.2, -0.2, -9.5, -10.8, -10.2, -8.2, -1.1, -1.3, -0.8, 1.2, -8.6, -7.8, -7.8, -5.0, 9.0, 9.5, 8.9, 12.0, -50.0, -13.0, -50.0, -10.3, -6.2, -7.0, -7.1, -5.4, -50.0, -9.3, -14.0, -8.2, -2.4, -6.3, -2.4, -3.9,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 33.3, -50.0, 30.6, -50.0, -50.0, -50.0, -50.0, -50.0, 29.2, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-14.8, -6.2, -14.9, -7.6, -14.6, -11.6, -13.0, -12.2, -14.0, -9.0, -14.1, -10.1, -13.6, -11.2, -12.4, -12.5, -8.1, 2.2, -8.7, 0.4, -15.3, -11.3, -13.2, -13.3, -11.9, -7.2, -11.8, -7.9, -10.7, -7.4, -10.8, -8.9, -15.5, -10.3, -15.5, -12.5, -14.5, -12.1, -11.9, -12.8, -18.3, -12.7, -16.4, -14.7, -14.2, -10.4, -13.7, -13.0, -50.0, 15.1, -50.0, 13.3, -9.9, -5.8, -8.9, -7.6, -50.0, -1.5, -7.8, -3.0, -8.1, 3.6, -9.4, 2.0,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 30.6, -50.0, 35.2, -50.0, -50.0, -50.0, -50.0, -50.0, 28.5, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-13.8, -7.5, -14.4, -5.1, -12.6, -11.6, -11.5, -10.1, -13.2, -9.7, -12.7, -7.4, -11.9, -11.9, -11.5, -10.2, -7.2, 0.9, -8.6, 2.6, -14.1, -12.8, -13.3, -11.0, -9.7, -8.4, -11.2, -5.3, -9.3, -8.7, -9.9, -7.1, -13.9, -11.2, -14.2, -8.9, -12.9, -12.6, -12.6, -10.9, -15.4, -13.8, -14.8, -11.4, -12.8, -11.6, -13.2, -10.3, -50.0, 13.3, -50.0, 15.2, -8.6, -7.0, -8.4, -4.8, -50.0, -3.0, -7.1, -0.1, -7.3, 2.3, -8.7, 3.9,],\n            [-7.3, -6.3, -8.9, -5.5, 2.9, 0.2, 1.5, 1.2, -8.5, -2.3, -10.0, -1.7, -5.6, -10.5, -5.5, -9.6, -4.3, -7.9, -6.1, -7.2, 2.3, -0.6, 0.5, -0.0, -8.9, -11.0, -9.4, -9.3, -4.6, -9.4, -7.3, -8.1, -7.8, -9.8, -8.9, -9.2, 2.8, 0.1, 1.2, 0.8, -5.9, -6.8, -5.9, -6.3, -4.1, -7.2, -6.1, -6.2, -50.0, -9.9, -50.0, -8.6, 12.5, 9.4, 11.0, 9.8, -50.0, -4.8, -8.7, -3.5, -0.0, -8.6, -2.1, -7.0,],\n            [-10.1, -4.3, -10.8, -5.8, 0.1, 2.3, 0.7, 0.6, -10.4, -0.2, -10.8, -2.0, -8.8, -9.2, -8.0, -9.7, -6.8, -5.9, -7.5, -7.0, -1.4, 2.2, -0.9, -0.4, -10.8, -7.7, -9.8, -8.8, -8.9, -7.3, -9.6, -9.1, -10.5, -8.3, -10.2, -10.0, -0.0, 2.5, 0.7, 0.5, -7.5, -5.1, -7.1, -6.7, -7.4, -5.7, -8.1, -7.0, -50.0, -5.8, -50.0, -7.0, 9.4, 12.7, 10.1, 10.0, -50.0, -1.5, -11.3, -3.3, -5.4, -4.2, -5.6, -6.8,],\n            [-8.4, -6.2, -8.9, -5.9, 1.8, 0.4, 2.7, 1.2, -9.1, -1.2, -8.8, -1.5, -7.2, -10.5, -4.9, -9.6, -5.7, -7.4, -5.5, -7.4, 0.3, -0.3, 1.8, -0.4, -9.0, -8.7, -7.3, -8.9, -5.9, -9.4, -6.4, -8.7, -8.7, -9.1, -8.5, -9.4, 1.4, 0.7, 3.5, 0.7, -6.6, -5.3, -4.7, -5.8, -5.7, -7.5, -5.5, -7.1, -50.0, -8.9, -50.0, -8.4, 11.0, 10.1, 13.2, 10.2, -50.0, -4.2, -6.1, -3.6, -3.5, -8.4, -1.2, -6.8,],\n            [-9.1, -5.4, -10.4, -4.2, 1.0, 0.5, 0.6, 2.4, -9.6, -1.5, -10.5, -0.5, -8.9, -10.5, -7.2, -8.3, -6.3, -6.9, -7.7, -5.1, -0.5, 0.1, -0.6, 2.3, -11.1, -10.2, -10.5, -7.0, -8.3, -9.4, -9.4, -6.6, -9.6, -9.7, -10.0, -7.9, 0.7, 0.6, 0.8, 2.3, -7.1, -6.5, -7.2, -5.0, -6.5, -7.4, -7.5, -5.4, -50.0, -7.6, -50.0, -4.8, 9.8, 10.0, 10.2, 12.1, -50.0, -3.1, -11.4, -0.7, -4.6, -6.6, -4.9, -3.9,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 29.2, -50.0, 28.5, -50.0, -50.0, -50.0, -50.0, -50.0, 33.3, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-13.0, -7.0, -13.1, -8.3, -9.2, -6.5, -8.9, -7.6, -9.8, -0.7, -9.2, -2.1, -12.4, -10.3, -12.4, -11.6, -9.5, -4.6, -10.4, -5.7, -13.6, -10.1, -11.4, -10.9, -6.5, -1.5, -6.2, -3.6, -12.8, -8.0, -12.3, -9.7, -16.5, -12.7, -15.6, -14.3, -8.7, -6.0, -7.1, -7.5, -9.5, -4.1, -8.4, -6.1, -11.0, -8.6, -11.3, -9.3, -50.0, -1.5, -50.0, -3.0, -4.8, -1.5, -4.2, -3.1, -50.0, 16.4, -5.1, 14.2, -10.1, -4.3, -10.0, -5.4,],\n            [-13.5, -16.3, -11.8, -15.5, -13.6, -15.3, -12.1, -16.1, -7.7, -13.3, -4.2, -12.6, -14.1, -16.2, -10.3, -15.4, -9.6, -11.6, -7.7, -9.8, -14.3, -17.5, -11.1, -17.2, -5.3, -7.1, -2.2, -7.1, -9.6, -11.9, -8.7, -11.4, -17.1, -20.6, -15.3, -19.9, -14.1, -14.8, -12.4, -16.5, -11.7, -12.8, -6.4, -13.4, -14.3, -15.7, -12.7, -14.0, -50.0, -7.8, -50.0, -7.1, -8.7, -11.3, -6.1, -11.4, -50.0, -5.1, 18.6, -4.8, -8.6, -8.1, -4.5, -7.3,],\n            [-12.4, -8.2, -13.0, -5.9, -7.7, -7.8, -7.4, -5.8, -8.8, -2.0, -9.3, -0.4, -11.6, -11.3, -11.6, -9.4, -8.9, -5.6, -10.0, -3.2, -11.5, -11.1, -10.5, -8.3, -6.3, -4.9, -6.7, -1.2, -10.6, -9.7, -11.3, -8.0, -14.8, -13.1, -15.5, -11.4, -7.5, -7.5, -7.1, -5.9, -8.4, -5.8, -8.1, -3.6, -9.9, -9.7, -10.1, -8.2, -50.0, -3.0, -50.0, -0.1, -3.5, -3.3, -3.6, -0.7, -50.0, 14.2, -4.8, 16.4, -8.9, -5.8, -8.9, -3.4,],\n            [-10.7, -13.2, -12.4, -11.6, -4.8, -8.4, -6.1, -7.2, -10.3, -11.8, -11.0, -11.6, 2.8, -1.7, 0.7, -1.1, -6.2, -9.7, -7.3, -8.4, -5.3, -8.6, -6.1, -8.3, -10.5, -11.2, -9.8, -10.2, 9.5, 6.5, 7.6, 6.8, -12.1, -16.0, -13.0, -16.2, -5.2, -8.6, -7.1, -7.8, -12.5, -14.4, -12.1, -13.0, 1.4, -3.0, -1.6, -2.4, -50.0, -8.1, -50.0, -7.3, -0.0, -5.4, -3.5, -4.6, -50.0, -10.1, -8.6, -8.9, 13.2, -0.9, 9.7, -0.3,],\n            [-18.1, -12.3, -19.4, -14.2, -12.2, -9.5, -11.5, -10.8, -17.9, -11.9, -18.2, -13.9, -6.4, -3.7, -6.8, -5.4, -12.8, -5.9, -13.6, -7.4, -13.7, -9.9, -12.9, -12.7, -16.2, -11.6, -15.2, -12.7, -3.2, 0.5, -3.3, -1.8, -18.9, -16.6, -18.8, -19.2, -12.5, -9.4, -10.6, -11.1, -17.3, -14.1, -16.2, -16.0, -7.9, -3.7, -7.8, -6.3, -50.0, 3.6, -50.0, 2.3, -8.6, -4.2, -8.4, -6.6, -50.0, -4.3, -8.1, -5.8, -0.9, 14.2, -1.8, 11.6,],\n            [-11.8, -13.1, -11.5, -12.0, -6.2, -8.3, -5.4, -7.4, -11.8, -12.3, -11.1, -11.4, 0.5, -2.5, 2.4, -1.6, -6.8, -9.9, -6.4, -8.9, -6.1, -8.6, -5.0, -7.8, -10.4, -11.6, -8.5, -9.9, 8.2, 6.4, 8.7, 6.9, -12.4, -17.0, -12.6, -15.8, -6.5, -8.5, -5.0, -7.8, -13.2, -13.6, -10.3, -13.7, -0.7, -3.2, -0.3, -2.4, -50.0, -9.4, -50.0, -8.7, -2.1, -5.6, -1.2, -4.9, -50.0, -10.0, -4.5, -8.9, 9.7, -1.8, 11.3, -0.9,],\n            [-17.2, -13.3, -17.8, -11.6, -11.0, -11.1, -10.4, -9.0, -16.0, -12.7, -16.1, -11.1, -5.2, -5.4, -6.0, -3.3, -11.8, -7.3, -13.3, -5.9, -12.7, -11.9, -11.9, -9.3, -15.2, -15.0, -14.9, -11.0, -2.8, -1.8, -2.9, 0.4, -17.4, -18.4, -18.3, -15.9, -10.7, -11.2, -10.6, -9.0, -15.4, -14.8, -15.8, -13.3, -6.8, -5.9, -7.2, -3.9, -50.0, 2.0, -50.0, 3.9, -7.0, -6.8, -6.8, -3.9, -50.0, -5.4, -7.3, -3.4, -0.3, 11.6, -0.9, 14.1,]];\n        this.TRANS = [[5, 0, 0, 4],\n            [0, 5, 4, 0],\n            [0, 4, 5, 0],\n            [4, 0, 0, 5]];\n        this.CONBLO = { 'A': 0, 'R': 1, 'N': 2, 'D': 3, 'C': 4, 'Q': 5, 'E': 6, 'G': 7, 'H': 8, 'I': 9, 'L': 10, 'K': 11, 'M': 12, 'F': 13, 'P': 14, 'S': 15, 'T': 16, 'W': 17, 'Y': 18, 'V': 19, 'B': 20, 'Z': 21, 'X': 22, '*': 23 };\n        this.CONNUCL = { 'A': 0, 'T': 1, 'C': 2, 'G': 3 };\n        this.CONSCHN = { 'AAA': 0, 'AAC': 1, 'AAG': 2, 'AAT': 3, 'ACA': 4, 'ACC': 5, 'ACG': 6, 'ACT': 7, 'AGA': 8, 'AGC': 9, 'AGG': 10, 'AGT': 11, 'ATA': 12, 'ATC': 13, 'ATG': 14, 'ATT': 15, 'CAA': 16, 'CAC': 17, 'CAG': 18, 'CAT': 19, 'CCA': 20, 'CCC': 21, 'CCG': 22, 'CCT': 23, 'CGA': 24, 'CGC': 25, 'CGG': 26, 'CGT': 27, 'CTA': 28, 'CTC': 29, 'CTG': 30, 'CTT': 31, 'GAA': 32, 'GAC': 33, 'GAG': 34, 'GAT': 35, 'GCA': 36, 'GCC': 37, 'GCG': 38, 'GCT': 39, 'GGA': 40, 'GGC': 41, 'GGG': 42, 'GGT': 43, 'GTA': 44, 'GTC': 45, 'GTG': 46, 'GTT': 47, 'TAA': 48, 'TAC': 49, 'TAG': 50, 'TAT': 51, 'TCA': 52, 'TCC': 53, 'TCG': 54, 'TCT': 55, 'TGA': 56, 'TGC': 57, 'TGG': 58, 'TGT': 59, 'TTA': 60, 'TTC': 61, 'TTG': 62, 'TTT': 63 };\n        this.setMethod(alignGrid_);\n        this.setSequences(seq1_, seq2_);\n        this.setGap(gap_);\n    }\n    fillNeedleman() {\n        for (let i = 0; i <= this.len1; i++) {\n            this.dpGrid[i][0].value = -i * this.gap;\n            this.dpGrid[i][0].parentI = i - 1;\n            this.dpGrid[i][0].parentJ = 0;\n        }\n        for (let i = 1; i <= this.len2; i++) {\n            this.dpGrid[0][i].value = -i * this.gap;\n            this.dpGrid[0][i].parentI = 0;\n            this.dpGrid[0][i].parentJ = i - 1;\n        }\n    }\n    fillOne(i, j) {\n        this.dpGrid[i][j].value = Math.max(this.dpGrid[i - 1][j - 1].value + this.scores[i - 1][j - 1], Math.max(this.dpGrid[i - 1][j].value - this.gap, this.dpGrid[i][j - 1].value - this.gap));\n        if (this.dpGrid[i][j].value == this.dpGrid[i - 1][j - 1].value + this.scores[i - 1][j - 1]) {\n            this.dpGrid[i][j].parentI = i - 1;\n            this.dpGrid[i][j].parentJ = j - 1;\n        }\n        else if (this.dpGrid[i][j].value == this.dpGrid[i - 1][j].value - this.gap) {\n            this.dpGrid[i][j].parentI = i - 1;\n            this.dpGrid[i][j].parentJ = j;\n        }\n        else {\n            this.dpGrid[i][j].parentI = i;\n            this.dpGrid[i][j].parentJ = j - 1;\n        }\n    }\n    gridFromName(path) {\n        switch (path) {\n            case 'BLOSUM45':\n                this.alignGrid = this.BLOSUM45;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM50':\n                this.alignGrid = this.BLOSUM50;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM62':\n                this.alignGrid = this.BLOSUM62;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM80':\n                this.alignGrid = this.BLOSUM80;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM90':\n                this.alignGrid = this.BLOSUM90;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM30':\n                this.alignGrid = this.PAM30;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM70':\n                this.alignGrid = this.PAM70;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM250':\n                this.alignGrid = this.PAM250;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'NUCLEOTIDES':\n                this.alignGrid = this.NUCLEOTIDES;\n                this.connections = this.CONNUCL;\n                this.step = 1;\n                break;\n            case 'TRANS':\n                this.alignGrid = this.TRANS;\n                this.connections = this.CONNUCL;\n                this.step = 1;\n                break;\n            case 'SCHNEIDER':\n                this.alignGrid = this.SCHNEIDER;\n                this.connections = this.CONSCHN;\n                this.step = 3;\n                break;\n        }\n        this.fillScores();\n    }\n    constructResult(i, j) {\n        let res1 = \"\", res2 = \"\", gapS = \"\";\n        let total = this.dpGrid[i][j].value;\n        for (let i = 0; i < this.step; i++) {\n            gapS += '|';\n        }\n        while (i > 0 || j > 0) {\n            if (this.dpGrid[i][j].parentI == i) {\n                res1 = gapS + res1;\n                res2 = this.seq2.substr((j - 1) * this.step, this.step) + res2;\n            }\n            else if (this.dpGrid[i][j].parentJ == j) {\n                res2 = gapS + res2;\n                res1 = this.seq1.substr((i - 1) * this.step, this.step) + res1;\n            }\n            else {\n                if (i - 1 >= 0)\n                    res1 = this.seq1.substr((i - 1) * this.step, this.step) + res1;\n                else\n                    res1 = gapS + res1;\n                if (j - 1 >= 0)\n                    res2 = this.seq2.substr((j - 1) * this.step, this.step) + res2;\n                else\n                    res2 = gapS + res2;\n            }\n            let tempI = i;\n            i = this.dpGrid[i][j].parentI;\n            j = this.dpGrid[tempI][j].parentJ;\n        }\n        return new Aligned(res1, res2, total);\n    }\n    fillScores() {\n        this.scores = [];\n        for (let i = 0; i < this.len1; i++) {\n            this.scores.push([]);\n            for (let j = 0; j < this.len2; j++) {\n                this.scores[i].push(this.alignGrid[this.connections[this.seq1.substr(i * this.step, this.step)]][this.connections[this.seq2.substr(j * this.step, this.step)]]);\n            }\n        }\n    }\n    typeRec() {\n        if (/[ATGC]*/.test(this.seq1) && /[ATGC]*/.test(this.seq2))\n            return \"NUCLEOTIDES\";\n        if (/[ARNDCQEGHILKMFPSTWYVBZX*]*/.test(this.seq1) && /[ARNDCQEGHILKMFPSTWYVBZX*]*/.test(this.seq2))\n            return \"BLOSUM62\";\n        throw Error(\"Can't recognize sequence type\");\n    }\n    needlemanWunch() {\n        this.dpGrid = [];\n        for (let i = 0; i < this.len1 + 1; i++) {\n            let row = [];\n            for (let j = 0; j < this.len2 + 1; j++)\n                row.push(new Cell());\n            this.dpGrid.push(row);\n        }\n        this.fillNeedleman();\n        for (let i = 1; i <= this.len1; i++) {\n            for (let j = 1; j <= this.len2; j++) {\n                this.fillOne(i, j);\n            }\n        }\n        return this.constructResult(this.len1, this.len2);\n    }\n    smithWaterman() {\n        this.dpGrid = [];\n        for (let i = 0; i < this.len1 + 1; i++) {\n            let row = [];\n            for (let j = 0; j < this.len2 + 1; j++)\n                row.push(new Cell());\n            this.dpGrid.push(row);\n        }\n        let bestVal = 0;\n        let bestI = 0, bestJ = 0;\n        for (let i = 1; i <= this.len1; i++) {\n            for (let j = 1; j <= this.len2; j++) {\n                this.fillOne(i, j);\n                if (this.dpGrid[i][j].value <= 0) {\n                    this.dpGrid[i][j].value = 0;\n                    this.dpGrid[i][j].parentI = 0;\n                    this.dpGrid[i][j].parentJ = 0;\n                }\n                else if (this.dpGrid[this.dpGrid[i][j].parentI][this.dpGrid[i][j].parentJ].value == 0) {\n                    this.dpGrid[i][j].parentI = 0;\n                    this.dpGrid[i][j].parentJ = 0;\n                }\n                if (this.dpGrid[i][j].value > bestVal) {\n                    bestVal = this.dpGrid[i][j].value;\n                    bestI = i;\n                    bestJ = j;\n                }\n            }\n        }\n        return this.constructResult(bestI, bestJ);\n    }\n    setSequences(seq1_, seq2_) {\n        this.seq1 = seq1_;\n        this.seq2 = seq2_;\n        this.len1 = this.seq1.length / this.step;\n        this.len2 = this.seq2.length / this.step;\n        this.fillScores();\n    }\n    setGap(gap_) {\n        this.gap = gap_;\n    }\n    setMethod(alignGrid_ = \"\") {\n        if (alignGrid_ == \"AUTO\") {\n            alignGrid_ = this.typeRec();\n        }\n        this.gridFromName(alignGrid_);\n    }\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nexport function getMacroMol(monomers) {\n    return __awaiter(this, void 0, void 0, function* () {\n        let result = [];\n        const moduleRdkit = yield grok.functions.call('Chem:getRdKitModule');\n        for (let i = 0; i < monomers.length; i++) {\n            for (let j = 0; j < monomers[i].length; j++) {\n                const mol = moduleRdkit.get_mol(monomers[i][j]['molfile']);\n                const a = mol.get_v3Kmolblock();\n                const indices = getIndices(monomers[i][j], a);\n                monomers[i][j]['indices'] = indices;\n                monomers[i][j]['molfile'] = yield rotateBackboneV3000(a, indices);\n                mol === null || mol === void 0 ? void 0 : mol.delete();\n            }\n            result.push(linkV3000(monomers[i]));\n        }\n        return result;\n    });\n}\nfunction getIndices(monomer, molV3000) {\n    const molfile = monomer[\"molfile\"];\n    let indexStart = molfile.indexOf('M  RGP', 0) + 8;\n    let indexEnd = molfile.indexOf('\\n', indexStart);\n    const indicesData = molfile.substring(indexStart, indexEnd).replaceAll('  ', ' ').replaceAll('  ', ' ');\n    let parsedData = indicesData.split(' ');\n    const remFirst = parsedData[2] == '1' ? parseInt(parsedData[1]) : parseInt(parsedData[3]);\n    const remLast = parsedData[2] == '2' ? parseInt(parsedData[1]) : parseInt(parsedData[3]);\n    const numbers = extractAtomsBondsNumbersV3000(molV3000);\n    let indexBonds = molV3000.indexOf('M  V30 BEGIN BOND'); // V3000 index for bonds\n    indexBonds = molV3000.indexOf('\\n', indexBonds);\n    indexStart = indexBonds;\n    indexEnd = indexBonds;\n    let first = 0;\n    let last = 0;\n    let remBondFirst = 0;\n    let remBondLast = 0;\n    for (let j = 0; j < numbers.nbond; j++) {\n        if (first == 0 || last == 0) {\n            indexStart = molV3000.indexOf('V30', indexStart) + 4;\n            indexEnd = molV3000.indexOf('\\n', indexStart);\n            const bondData = molV3000.substring(indexStart, indexEnd).replaceAll('  ', ' ').replaceAll('  ', ' ');\n            parsedData = bondData.split(' ');\n            if (parseInt(parsedData[2]) == remFirst) {\n                first = parseInt(parsedData[3]);\n                remBondFirst = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[3]) == remFirst) {\n                first = parseInt(parsedData[2]);\n                remBondFirst = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[2]) == remLast) {\n                last = parseInt(parsedData[3]);\n                remBondLast = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[3]) == remLast) {\n                last = parseInt(parsedData[2]);\n                remBondLast = parseInt(parsedData[0]);\n            }\n        }\n    }\n    return { first, last, remFirst, remLast, remBondFirst, remBondLast };\n}\nfunction rotateBackboneV3000(molBlock, indices) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const coordinates = extractAtomDataV3000(molBlock);\n        const natom = coordinates.atomIndex.length;\n        const first = indices['first'];\n        const last = indices['last'];\n        const xCenter = (coordinates.x[last] + coordinates.x[first]) / 2;\n        const yCenter = (coordinates.y[last] + coordinates.y[first]) / 2;\n        //place to center\n        for (let i = 0; i < natom; i++) {\n            coordinates.x[i] -= xCenter;\n            coordinates.y[i] -= yCenter;\n        }\n        let angle = 0;\n        if (coordinates.x[first] == 0)\n            angle = coordinates.y[first] > coordinates.y[last] ? Math.PI / 2 : 3 * Math.PI / 2;\n        else if (coordinates.y[first] == 0)\n            angle = coordinates.x[first] > coordinates.x[last] ? Math.PI : 0;\n        else {\n            const derivative = coordinates.y[first] / coordinates.x[first];\n            if (coordinates.x[first] < coordinates.x[last])\n                angle = derivative > 0 ? Math.PI - Math.atan(derivative) : Math.atan(derivative);\n            else\n                angle = derivative > 0 ? Math.atan(derivative) : Math.PI - Math.atan(derivative);\n        }\n        const cos = Math.cos(angle);\n        const sin = Math.sin(angle);\n        for (let i = 0; i < natom; i++) {\n            const xAdd = coordinates.x[i];\n            coordinates.x[i] = xAdd * cos - coordinates.y[i] * sin;\n            coordinates.y[i] = xAdd * sin + coordinates.y[i] * cos;\n        }\n        //place to right\n        const xShift = coordinates.x[first];\n        for (let i = 0; i < natom; i++)\n            coordinates.x[i] -= xShift;\n        //rewrite molBlock\n        let index = molBlock.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n        index = molBlock.indexOf('\\n', index);\n        let indexEnd = index;\n        for (let i = 0; i < natom; i++) {\n            index = molBlock.indexOf('V30', index) + 4;\n            index = molBlock.indexOf(' ', index) + 1;\n            index = molBlock.indexOf(' ', index) + 1;\n            indexEnd = molBlock.indexOf(' ', index) + 1;\n            indexEnd = molBlock.indexOf(' ', indexEnd);\n            molBlock = molBlock.slice(0, index) +\n                coordinates.x[i] + ' ' + coordinates.y[i] +\n                molBlock.slice(indexEnd);\n            index = molBlock.indexOf('\\n', index) + 1;\n        }\n        return molBlock;\n    });\n}\nfunction extractAtomDataV3000(molBlock) {\n    const numbers = extractAtomsBondsNumbersV3000(molBlock);\n    let index = molBlock.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n    index = molBlock.indexOf('\\n', index);\n    let indexEnd = index;\n    const indexes = Array(numbers.natom);\n    const types = Array(numbers.natom);\n    const x = Array(numbers.natom);\n    const y = Array(numbers.natom);\n    for (let i = 0; i < numbers.natom; i++) {\n        index = molBlock.indexOf('V30', index) + 4;\n        indexEnd = molBlock.indexOf(' ', index);\n        indexes[i] = parseInt(molBlock.substring(index, indexEnd));\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        types[i] = molBlock.substring(index, indexEnd);\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        x[i] = parseFloat(molBlock.substring(index, indexEnd));\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        y[i] = parseFloat(molBlock.substring(index, indexEnd));\n        index = molBlock.indexOf('\\n', index) + 1;\n    }\n    return { atomIndex: indexes, atomType: types, x: x, y: y };\n}\nfunction linkV3000(monomers) {\n    let macroMolBlock = '\\nDatagrok macromolecule handler\\n\\n';\n    macroMolBlock += '  0  0  0  0  0  0            999 V3000\\n';\n    macroMolBlock += 'M  V30 BEGIN CTAB\\n';\n    let atomBlock = '';\n    let bondBlock = '';\n    let collectionBlock = '';\n    const collection = [];\n    let natom = 0;\n    let nbond = 0;\n    let xShift = 0;\n    for (let i = 0; i < monomers.length; i++) {\n        let molfile = monomers[i]['molfile'];\n        const first = monomers[i]['indices']['first'];\n        const last = monomers[i]['indices']['last'];\n        const remFirst = monomers[i]['indices']['remFirst'];\n        const remLast = monomers[i]['indices']['remLast'];\n        const remBondFirst = monomers[i]['indices']['remBondFirst'];\n        const remBondLast = monomers[i]['indices']['remBondLast'];\n        molfile = molfile.replaceAll('(-\\nM  V30 ', '(')\n            .replaceAll('-\\nM  V30 ', '').replaceAll(' )', ')');\n        const numbers = extractAtomsBondsNumbersV3000(molfile);\n        const coordinates = extractAtomDataV3000(molfile);\n        let indexAtoms = molfile.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n        indexAtoms = molfile.indexOf('\\n', indexAtoms);\n        let index = indexAtoms;\n        let indexEnd = indexAtoms;\n        const totalShift = xShift - coordinates.x[first - 1];\n        for (let j = 0; j < numbers.natom; j++) {\n            if (coordinates.atomIndex[j] != remFirst && coordinates.atomIndex[j] != remLast) { //|| i == 0) {\n                //rewrite atom number\n                index = molfile.indexOf('V30', index) + 4;\n                indexEnd = molfile.indexOf(' ', index);\n                let atomNumber = parseInt(molfile.substring(index, indexEnd));\n                atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                    (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n                atomNumber += natom;\n                molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n                //rewrite coordinates\n                index = molfile.indexOf(' ', index) + 1;\n                index = molfile.indexOf(' ', index) + 1;\n                indexEnd = molfile.indexOf(' ', index);\n                let coordinate = Math.round(10000 * (parseFloat(molfile.substring(index, indexEnd)) + totalShift)) / 10000;\n                molfile = molfile.slice(0, index) + coordinate + molfile.slice(indexEnd);\n                index = molfile.indexOf(' ', index) + 1;\n                indexEnd = molfile.indexOf(' ', index);\n                coordinate = Math.round(10000 * (parseFloat(molfile.substring(index, indexEnd)))) / 10000;\n                molfile = molfile.slice(0, index) + coordinate + molfile.slice(indexEnd);\n                index = molfile.indexOf('\\n', index) + 1;\n            }\n            else {\n                index = molfile.indexOf('M  V30', index) - 1;\n                indexEnd = molfile.indexOf('\\n', index + 1);\n                molfile = molfile.slice(0, index) + molfile.slice(indexEnd);\n            }\n        }\n        const indexAtomsEnd = molfile.indexOf('M  V30 END ATOM');\n        atomBlock += molfile.substring(indexAtoms + 1, indexAtomsEnd);\n        let indexBonds = molfile.indexOf('M  V30 BEGIN BOND'); // V3000 index for bonds\n        indexBonds = molfile.indexOf('\\n', indexBonds);\n        index = indexBonds;\n        indexEnd = indexBonds;\n        let bondNumber = 0;\n        for (let j = 0; j < numbers.nbond; j++) {\n            //rewrite bond number\n            index = molfile.indexOf('V30', index) + 4;\n            indexEnd = molfile.indexOf(' ', index);\n            bondNumber = parseInt(molfile.substring(index, indexEnd));\n            if (bondNumber == remBondFirst || bondNumber == remBondLast) {\n                indexEnd = molfile.indexOf('\\n', index) + 1;\n                index -= 7;\n                molfile = molfile.slice(0, index) + molfile.slice(indexEnd);\n                continue;\n            }\n            bondNumber = (bondNumber > remBondFirst && bondNumber > remBondLast) ? bondNumber - 2 :\n                (bondNumber > remBondFirst || bondNumber > remBondLast) ? bondNumber - 1 : bondNumber;\n            bondNumber += nbond;\n            molfile = molfile.slice(0, index) + bondNumber + molfile.slice(indexEnd);\n            //rewrite atom pair in bond\n            index = molfile.indexOf(' ', index) + 1;\n            index = molfile.indexOf(' ', index) + 1;\n            indexEnd = molfile.indexOf(' ', index);\n            let atomNumber = parseInt(molfile.substring(index, indexEnd));\n            atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n            atomNumber += natom;\n            molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n            index = molfile.indexOf(' ', index) + 1;\n            indexEnd = Math.min(molfile.indexOf('\\n', index), molfile.indexOf(' ', index));\n            atomNumber = parseInt(molfile.substring(index, indexEnd));\n            atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n            atomNumber += natom;\n            molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n            index = molfile.indexOf('\\n', index) + 1;\n        }\n        const indexBondEnd = molfile.indexOf('M  V30 END BOND');\n        bondBlock += molfile.substring(indexBonds + 1, indexBondEnd);\n        //let indexCollection = molfile.indexOf('M  V30 MDLV30/STEABS ATOMS=('); // V3000 index for collections\n        // while (indexCollection != -1) {\n        //   indexCollection += 28;\n        //   const collectionEnd = molfile.indexOf(')', indexCollection);\n        //   const collectionEntries = molfile.substring(indexCollection, collectionEnd).split(' ').slice(1);\n        //   collectionEntries.forEach((e: string) => {\n        //     collection.push(parseInt(e) + natom);\n        //   });\n        //   indexCollection = collectionEnd;\n        //   indexCollection = molfile.indexOf('M  V30 MDLV30/STEABS ATOMS=(', indexCollection);\n        // }\n        natom += numbers.natom - 2;\n        nbond += numbers.nbond - 2;\n        xShift += coordinates.x[last] - coordinates.x[first] + 1;\n        if (i == monomers.length - 1) {\n            natom++;\n            const shift = xShift + 0.2;\n            atomBlock += 'M  V30 ' + natom + ' O ' + shift + ' 0 0.000000 0\\n';\n        }\n        nbond++;\n        if (i == monomers.length - 1) {\n            const rightTerminal = (last > remFirst && last > remLast) ? last + natom - (numbers.natom - 2) - 3 :\n                (last > remFirst || last > remLast) ? last + natom - (numbers.natom - 2) - 2 :\n                    last + natom - (numbers.natom - 2) - 1;\n            bondBlock += 'M  V30 ' + nbond + ' 1 ' + rightTerminal + ' ' + natom + '\\n';\n        }\n        else {\n            const rightTerminal = (last > remFirst && last > remLast) ? last + natom - (numbers.natom - 2) - 2 :\n                (last > remFirst || last > remLast) ? last + natom - (numbers.natom - 2) - 1 :\n                    last + natom - (numbers.natom - 2);\n            const next = monomers[i + 1]['indices'];\n            const nextFirst = next['first'];\n            const nextRemFirst = next['remFirst'];\n            const nextRemLast = next['remLast'];\n            const leftTerminal = (nextFirst > nextRemFirst && nextFirst > nextRemLast) ? nextFirst + natom - 2 :\n                (nextFirst > nextRemFirst || nextFirst > nextRemLast) ? nextFirst + natom - 1 :\n                    nextFirst + natom;\n            bondBlock += 'M  V30 ' + nbond + ' 1 ' + rightTerminal + ' ' + leftTerminal + '\\n';\n        }\n    }\n    // const entries = 4;\n    // const collNumber = Math.ceil(collection.length / entries);\n    // collectionBlock += 'M  V30 MDLV30/STEABS ATOMS=(' + collection.length + ' -\\n';\n    // for (let i = 0; i < collNumber; i++) {\n    //   collectionBlock += 'M  V30 ';\n    //   const entriesCurrent = i + 1 == collNumber ? collection.length - (collNumber - 1)*entries : entries;\n    //   for (let j = 0; j < entriesCurrent; j++) {\n    //     collectionBlock += (j + 1 == entriesCurrent) ?\n    //       (i == collNumber - 1 ? collection[entries*i + j] + ')\\n' : collection[entries*i + j] + ' -\\n') :\n    //       collection[entries*i + j] + ' ';\n    //   }\n    // }\n    //generate file\n    macroMolBlock += 'M  V30 COUNTS ' + natom + ' ' + nbond + ' 0 0 0\\n';\n    macroMolBlock += 'M  V30 BEGIN ATOM\\n';\n    macroMolBlock += atomBlock;\n    macroMolBlock += 'M  V30 END ATOM\\n';\n    macroMolBlock += 'M  V30 BEGIN BOND\\n';\n    macroMolBlock += bondBlock;\n    macroMolBlock += 'M  V30 END BOND\\n';\n    //macroMolBlock += 'M  V30 BEGIN COLLECTION\\n';\n    //macroMolBlock += collectionBlock;\n    //macroMolBlock += 'M  V30 END COLLECTION\\n';\n    macroMolBlock += 'M  V30 END CTAB\\n';\n    macroMolBlock += 'M  END\\n';\n    return macroMolBlock;\n}\nfunction extractAtomsBondsNumbersV3000(molBlock) {\n    molBlock = molBlock.replaceAll('\\r', ''); //equalize old and new sdf standards\n    let index = molBlock.indexOf('COUNTS') + 7; // V3000 index for atoms and bonds number\n    let indexEnd = molBlock.indexOf(' ', index);\n    const atomsNumber = parseInt(molBlock.substring(index, indexEnd));\n    index = indexEnd + 1;\n    indexEnd = molBlock.indexOf(' ', index);\n    const bondsNumber = parseInt(molBlock.substring(index, indexEnd));\n    return { natom: atomsNumber, nbond: bondsNumber };\n}\n","import * as C from './constants';\nimport * as DG from 'datagrok-api/dg';\nimport { AminoacidsPalettes } from '@datagrok-libraries/bio/src/aminoacids';\nimport { NucleotidesPalettes } from '@datagrok-libraries/bio/src/nucleotides';\nimport { UnknownSeqPalettes } from '@datagrok-libraries/bio/src/unknown';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport * as ui from 'datagrok-api/ui';\nimport { printLeftOrCentered, DrawStyle } from '@datagrok-libraries/bio/src/utils/cell-renderer';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nconst undefinedColor = 'rgb(100,100,100)';\nconst monomerToShortFunction = WebLogo.monomerToShort;\nconst gapRenderer = 5;\nfunction getPaletteByType(paletteType) {\n    switch (paletteType) {\n        case 'PT':\n            return AminoacidsPalettes.GrokGroups;\n        case 'NT':\n            return NucleotidesPalettes.Chromatogram;\n        case 'DNA':\n            return NucleotidesPalettes.Chromatogram;\n        case 'RNA':\n            return NucleotidesPalettes.Chromatogram;\n        // other\n        default:\n            return UnknownSeqPalettes.Color;\n    }\n}\nfunction getUpdatedWidth(grid, g, x, w) {\n    return grid ? Math.min(grid.canvas.width - x, w) : g.canvas.width - x;\n}\nexport function processSequence(subParts) {\n    const simplified = !subParts.some((amino, index) => amino.length > 1 &&\n        index != 0 &&\n        index != subParts.length - 1);\n    const text = [];\n    const gap = simplified ? '' : ' ';\n    subParts.forEach((amino, index) => {\n        if (index < subParts.length)\n            amino += `${amino ? '' : '-'}${gap}`;\n        text.push(amino);\n    });\n    return [text, simplified];\n}\nexport class MacromoleculeSequenceCellRenderer extends DG.GridCellRenderer {\n    get name() { return 'sequence'; }\n    get cellType() { return 'sequence'; }\n    get defaultHeight() { return 30; }\n    get defaultWidth() { return 230; }\n    onMouseMove(gridCell, e) {\n        var _a, _b, _c;\n        if (gridCell.cell.column.getTag(UnitsHandler.TAGS.aligned) !== 'SEQ.MSA') {\n            return;\n        }\n        const maxLengthWordsSum = gridCell.cell.column.temp['bio-sum-maxLengthWords'];\n        const maxIndex = gridCell.cell.column.temp['bio-maxIndex'];\n        const argsX = e.offsetX - gridCell.gridColumn.left + (gridCell.gridColumn.left - gridCell.bounds.x);\n        let left = 0;\n        let right = maxIndex;\n        let found = false;\n        maxLengthWordsSum[maxIndex + 1] = argsX + 1;\n        let mid = 0;\n        if (argsX > maxLengthWordsSum[0]) {\n            while (!found) {\n                mid = Math.floor((right + left) / 2);\n                if (argsX >= maxLengthWordsSum[mid] && argsX <= maxLengthWordsSum[mid + 1]) {\n                    left = mid;\n                    found = true;\n                }\n                else if (argsX < maxLengthWordsSum[mid]) {\n                    right = mid - 1;\n                }\n                else if (argsX > maxLengthWordsSum[mid + 1]) {\n                    left = mid + 1;\n                }\n                if (left == right) {\n                    found = true;\n                }\n            }\n        }\n        left = (argsX >= maxLengthWordsSum[left]) ? left + 1 : left;\n        const separator = (_a = gridCell.cell.column.getTag('separator')) !== null && _a !== void 0 ? _a : '';\n        const splitterFunc = WebLogo.getSplitter('separator', separator);\n        const subParts = splitterFunc(gridCell.cell.value);\n        ((((_c = (_b = subParts[left]) === null || _b === void 0 ? void 0 : _b.length) !== null && _c !== void 0 ? _c : 0) > 0)) ? ui.tooltip.show(ui.div(subParts[left]), e.x + 16, e.y + 16) : ui.tooltip.hide();\n    }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     * @memberof AlignedSequenceCellRenderer\n     */\n    render(g, x, y, w, h, gridCell, cellStyle) {\n        var _a;\n        const grid = gridCell.gridRow !== -1 ? gridCell.grid : null;\n        const cell = gridCell.cell;\n        const paletteType = gridCell.cell.column.getTag(C.TAGS.ALPHABET);\n        const minDistanceRenderer = 50;\n        w = getUpdatedWidth(grid, g, x, w);\n        g.save();\n        g.beginPath();\n        g.rect(x, y, w, h);\n        g.clip();\n        g.font = '12px monospace';\n        g.textBaseline = 'top';\n        //TODO: can this be replaced/merged with splitSequence?\n        const units = gridCell.cell.column.getTag(DG.TAGS.UNITS);\n        const palette = getPaletteByType(paletteType);\n        const separator = (_a = gridCell.cell.column.getTag('separator')) !== null && _a !== void 0 ? _a : '';\n        const splitLimit = gridCell.bounds.width / 5;\n        const splitterFunc = WebLogo.getSplitter(units, separator, gridCell.bounds.width / 5);\n        const maxLengthOfMonomer = 8;\n        let maxLengthWords = {};\n        if (gridCell.cell.column.getTag('.calculatedCellRender') !== splitLimit.toString()) {\n            let samples = 0;\n            while (samples < Math.min(gridCell.cell.column.length, 100)) {\n                let column = gridCell.cell.column.get(samples);\n                let subParts = splitterFunc(column);\n                subParts.forEach((amino, index) => {\n                    var _a, _b;\n                    let textSize = monomerToShortFunction(amino, maxLengthOfMonomer).length * 7 + gapRenderer;\n                    if (textSize > ((_a = maxLengthWords[index]) !== null && _a !== void 0 ? _a : 0)) {\n                        maxLengthWords[index] = textSize;\n                    }\n                    if (index > ((_b = maxLengthWords['bio-maxIndex']) !== null && _b !== void 0 ? _b : 0)) {\n                        maxLengthWords['bio-maxIndex'] = index;\n                    }\n                });\n                samples += 1;\n            }\n            let minLength = 3 * 7;\n            for (let i = 0; i <= maxLengthWords['bio-maxIndex']; i++) {\n                if (maxLengthWords[i] < minLength) {\n                    maxLengthWords[i] = minLength;\n                }\n            }\n            let maxLengthWordSum = {};\n            maxLengthWordSum[0] = maxLengthWords[0];\n            for (let i = 1; i <= maxLengthWords['bio-maxIndex']; i++) {\n                maxLengthWordSum[i] = maxLengthWordSum[i - 1] + maxLengthWords[i];\n            }\n            gridCell.cell.column.temp = {\n                'bio-sum-maxLengthWords': maxLengthWordSum,\n                'bio-maxIndex': maxLengthWords['bio-maxIndex'],\n                'bio-maxLengthWords': maxLengthWords\n            };\n            gridCell.cell.column.setTag('.calculatedCellRender', splitLimit.toString());\n        }\n        else {\n            maxLengthWords = gridCell.cell.column.temp['bio-maxLengthWords'];\n        }\n        const subParts = splitterFunc(cell.value);\n        let x1 = x;\n        let color = undefinedColor;\n        let drawStyle = DrawStyle.classic;\n        if (gridCell.cell.column.getTag('aligned').includes('MSA') && gridCell.cell.column.getTag('units') === 'separator') {\n            drawStyle = DrawStyle.MSA;\n        }\n        subParts.every((amino, index) => {\n            color = palette.get(amino);\n            g.fillStyle = undefinedColor;\n            let last = index === subParts.length - 1;\n            x1 = printLeftOrCentered(x1, y, w, h, g, monomerToShortFunction(amino, maxLengthOfMonomer), color, 0, true, 1.0, separator, last, drawStyle, maxLengthWords, index, gridCell);\n            return x1 - minDistanceRenderer - gridCell.gridColumn.left + (gridCell.gridColumn.left - gridCell.bounds.x) <= gridCell.bounds.width;\n        });\n        g.restore();\n        return;\n    }\n}\nexport class MonomerCellRenderer extends DG.GridCellRenderer {\n    get name() { return C.SEM_TYPES.MONOMER; }\n    get cellType() { return C.SEM_TYPES.MONOMER; }\n    get defaultHeight() { return 15; }\n    get defaultWidth() { return 30; }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     */\n    render(g, x, y, w, h, gridCell, _cellStyle) {\n        g.font = `12px monospace`;\n        g.textBaseline = 'middle';\n        g.textAlign = 'center';\n        const palette = getPaletteByType(gridCell.cell.column.getTag(C.TAGS.ALPHABET));\n        const s = gridCell.cell.value ? gridCell.cell.value : '-';\n        const color = palette.get(s);\n        g.fillStyle = color;\n        g.fillText(s, x + (w / 2), y - (h / 2));\n    }\n}\nexport class MacromoleculeDifferenceCellRenderer extends DG.GridCellRenderer {\n    get name() { return 'MacromoleculeDifferenceCR'; }\n    get cellType() { return C.SEM_TYPES.MACROMOLECULE_DIFFERENCE; }\n    get defaultHeight() { return 30; }\n    get defaultWidth() { return 230; }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     * @memberof AlignedSequenceDifferenceCellRenderer\n     */\n    render(g, x, y, w, h, gridCell, cellStyle) {\n        var _a;\n        const grid = gridCell.grid;\n        const cell = gridCell.cell;\n        w = getUpdatedWidth(grid, g, w, x);\n        g.save();\n        g.beginPath();\n        g.rect(x, y, w, h);\n        g.clip();\n        g.font = '12px monospace';\n        g.textBaseline = 'top';\n        const s = (_a = cell.value) !== null && _a !== void 0 ? _a : '';\n        //TODO: can this be replaced/merged with splitSequence?\n        const [s1, s2] = s.split('#');\n        const separator = gridCell.tableColumn.tags[C.TAGS.SEPARATOR];\n        const units = gridCell.tableColumn.tags[DG.TAGS.UNITS];\n        const splitter = WebLogo.getSplitter(units, separator);\n        const subParts1 = splitter(s1);\n        const subParts2 = splitter(s2);\n        const [text] = processSequence(subParts1);\n        const textSize = g.measureText(text.join(''));\n        let updatedX = Math.max(x, x + (w - (textSize.width + subParts1.length * 4)) / 2);\n        // 28 is the height of the two substitutions on top of each other + space\n        const updatedY = Math.max(y, y + (h - 28) / 2);\n        let palette = UnknownSeqPalettes.Color;\n        if (units != 'HELM')\n            palette = getPaletteByType(units.substring(units.length - 2));\n        const vShift = 7;\n        for (let i = 0; i < subParts1.length; i++) {\n            const amino1 = subParts1[i];\n            const amino2 = subParts2[i];\n            const color1 = palette.get(amino1);\n            if (amino1 != amino2) {\n                const color2 = palette.get(amino2);\n                const subX0 = printLeftOrCentered(updatedX, updatedY - vShift, w, h, g, amino1, color1, 0, true);\n                const subX1 = printLeftOrCentered(updatedX, updatedY + vShift, w, h, g, amino2, color2, 0, true);\n                updatedX = Math.max(subX1, subX0);\n            }\n            else\n                updatedX = printLeftOrCentered(updatedX, updatedY, w, h, g, amino1, color1, 0, true, 0.5);\n            updatedX += 4;\n        }\n        g.restore();\n    }\n}\n","export var COLUMNS_NAMES;\n(function (COLUMNS_NAMES) {\n    COLUMNS_NAMES[\"SPLIT_COL\"] = \"~split\";\n    COLUMNS_NAMES[\"ACTIVITY\"] = \"~activity\";\n    COLUMNS_NAMES[\"ACTIVITY_SCALED\"] = \"activity_scaled\";\n    COLUMNS_NAMES[\"ALIGNED_SEQUENCE\"] = \"~aligned_sequence\";\n    COLUMNS_NAMES[\"AMINO_ACID_RESIDUE\"] = \"AAR\";\n    COLUMNS_NAMES[\"POSITION\"] = \"Pos\";\n    COLUMNS_NAMES[\"P_VALUE\"] = \"pValue\";\n    COLUMNS_NAMES[\"MEAN_DIFFERENCE\"] = \"Mean difference\";\n})(COLUMNS_NAMES || (COLUMNS_NAMES = {}));\nexport var TAGS;\n(function (TAGS) {\n    TAGS[\"AAR\"] = \"AAR\";\n    TAGS[\"POSITION\"] = \"Pos\";\n    TAGS[\"SEPARATOR\"] = \"separator\";\n    TAGS[\"SELECTION\"] = \"selection\";\n    TAGS[\"ALPHABET\"] = \"alphabet\";\n    TAGS[\"ALIGNED\"] = \"aligned\";\n})(TAGS || (TAGS = {}));\nexport var SEM_TYPES;\n(function (SEM_TYPES) {\n    SEM_TYPES[\"MONOMER\"] = \"Monomer\";\n    SEM_TYPES[\"MACROMOLECULE_DIFFERENCE\"] = \"MacromoleculeDifference\";\n    SEM_TYPES[\"ACTIVITY\"] = \"activity\";\n    SEM_TYPES[\"ACTIVITY_SCALED\"] = \"activityScaled\";\n    SEM_TYPES[\"MACROMOLECULE\"] = \"Macromolecule\";\n})(SEM_TYPES || (SEM_TYPES = {}));\nexport const MSA = 'MSA';\nexport const STATS = 'stats';\nexport const EMBEDDING_STATUS = 'embeddingStatus';\nexport const PEPTIDES_ANALYSIS = 'isPeptidesAnalysis';\nexport var FLAGS;\n(function (FLAGS) {\n    FLAGS[\"CELL_CHANGING\"] = \"isCellChanging\";\n})(FLAGS || (FLAGS = {}));\nexport const aarGroups = {\n    'R': 'PC', 'H': 'PC', 'K': 'PC',\n    'D': 'NC', 'E': 'NC',\n    'S': 'U', 'T': 'U', 'N': 'U', 'Q': 'U',\n    'C': 'SC', 'U': 'SC', 'G': 'SC', 'P': 'SC',\n    'A': 'H', 'V': 'H', 'I': 'H', 'L': 'H', 'M': 'H', 'F': 'H', 'Y': 'H', 'W': 'H',\n    '-': '-',\n};\nexport const groupDescription = {\n    'PC': { 'description': 'Positive Amino Acids, with Electrically Charged Side Chains', 'aminoAcids': ['R', 'H', 'K'] },\n    'NC': { 'description': 'Negative Amino Acids, with Electrically Charged Side Chains', 'aminoAcids': ['D', 'E'] },\n    'U': { 'description': 'Amino Acids with Polar Uncharged Side Chains', 'aminoAcids': ['S', 'T', 'N', 'Q'] },\n    'SC': { 'description': 'Special Cases', 'aminoAcids': ['C', 'U', 'G', 'P'] },\n    'H': {\n        'description': 'Amino Acids with Hydrophobic Side Chain',\n        'aminoAcids': ['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'],\n    },\n    '-': { 'description': 'Unknown Amino Acid', 'aminoAcids': ['-'] },\n};\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as ui from 'datagrok-api/ui';\nimport * as grok from 'datagrok-api/grok';\nimport $ from 'cash-dom';\nimport { NotationConverter } from '@datagrok-libraries/bio/src/utils/notation-converter';\nlet convertDialog = null;\nlet convertDialogSubs = [];\n/**\n * Converts notations of a Macromolecule column\n *\n * @param {DG.column} col Column with 'Macromolecule' semantic type\n */\nexport function convert(col) {\n    const converter = new NotationConverter(col);\n    const currentNotation = converter.notation;\n    //TODO: read all notations\n    const notations = [\n        \"fasta\" /* NOTATION.FASTA */,\n        \"separator\" /* NOTATION.SEPARATOR */,\n        \"helm\" /* NOTATION.HELM */\n    ];\n    const separatorArray = ['-', '.', '/'];\n    const filteredNotations = notations.filter((e) => e !== currentNotation);\n    const targetNotationInput = ui.choiceInput('Convert to', filteredNotations[0], filteredNotations);\n    const separatorInput = ui.choiceInput('Separator', separatorArray[0], separatorArray);\n    // hide the separator input for non-SEPARATOR target notations\n    const toggleSeparator = () => {\n        if (targetNotationInput.value !== \"separator\" /* NOTATION.SEPARATOR */)\n            $(separatorInput.root).hide();\n        else\n            $(separatorInput.root).show();\n    };\n    // set correct visibility on init\n    toggleSeparator();\n    targetNotationInput.onChanged(() => {\n        toggleSeparator();\n    });\n    if (convertDialog == null) {\n        convertDialog = ui.dialog('Convert sequence notation')\n            .add(ui.div([\n            ui.divText('Current notation: ' + currentNotation, {\n                style: {\n                    'text-align': 'center',\n                    'font-weight': 'bold',\n                    'font-size': '14px',\n                    'padding': '5px',\n                }\n            }),\n            targetNotationInput.root,\n            separatorInput.root\n        ]))\n            .onOK(() => __awaiter(this, void 0, void 0, function* () {\n            const targetNotation = targetNotationInput.value;\n            const separator = separatorInput.value;\n            yield convertDo(col, targetNotation, separator);\n        }))\n            .show({ x: 350, y: 100 });\n        convertDialogSubs.push(convertDialog.onClose.subscribe((value) => {\n            convertDialogSubs.forEach((s) => { s.unsubscribe(); });\n            convertDialogSubs = [];\n            convertDialog = null;\n        }));\n    }\n}\nexport function convertDo(srcCol, targetNotation, separator) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const converter = new NotationConverter(srcCol);\n        const newColumn = converter.convert(targetNotation, separator);\n        srcCol.dataFrame.columns.add(newColumn);\n        // call to calculate 'cell.renderer' tag\n        yield grok.data.detectSemanticTypes(srcCol.dataFrame);\n        return newColumn;\n    });\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport { FastaFileHandler } from '@datagrok-libraries/bio/src/utils/fasta-handler';\n//@ts-ignore\nimport Aioli from '@biowasm/aioli';\nimport { AlignedSequenceEncoder } from '@datagrok-libraries/bio/src/sequence-encoder';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\n/**\n * Converts array of sequences into simple fasta string.\n *\n * @param {string[]} sequences Input list of sequences.\n * @return {string} Fasta-formatted string.\n */\nfunction _stringsToFasta(sequences) {\n    return sequences.reduce((a, v, i) => a + `>sample${i + 1}\\n${v}\\n`, '');\n}\n/**\n * Runs Aioli environment with kalign tool.\n *\n * @param {DG.Column} srcCol Column with sequences.\n * @param {boolean} isAligned Whether the column is aligned.\n * @param {string} unUsedName\n * @return {Promise<DG.Column>} Aligned sequences.\n */\nexport function runKalign(srcCol, isAligned = false, unUsedName = '') {\n    return __awaiter(this, void 0, void 0, function* () {\n        let sequences = srcCol.toList();\n        if (isAligned)\n            sequences = sequences.map((v, _) => AlignedSequenceEncoder.clean(v).replace(/\\-/g, ''));\n        const fasta = _stringsToFasta(sequences);\n        const CLI = yield new Aioli({\n            tool: 'kalign',\n            version: '3.3.1',\n            reinit: true,\n        });\n        console.log(['fasta.length =', fasta.length]);\n        yield CLI.fs.writeFile('input.fa', fasta);\n        const output = yield CLI.exec('kalign input.fa -f fasta -o result.fasta');\n        console.warn(output);\n        const buf = yield CLI.cat('result.fasta');\n        if (!buf)\n            throw new Error(`kalign output no result`);\n        const ffh = new FastaFileHandler(buf);\n        const aligned = ffh.sequencesArray; // array of sequences extracted from FASTA\n        const tgtCol = DG.Column.fromStrings(unUsedName, aligned);\n        // units\n        const srcUnits = srcCol.getTag(DG.TAGS.UNITS);\n        //aligned\n        const srcAligned = srcCol.getTag(UnitsHandler.TAGS.aligned);\n        const tgtAligned = srcAligned + '.MSA';\n        //alphabet\n        const srcAlphabet = srcCol.getTag(UnitsHandler.TAGS.alphabet);\n        tgtCol.setTag(DG.TAGS.UNITS, srcUnits);\n        tgtCol.setTag(UnitsHandler.TAGS.aligned, tgtAligned);\n        tgtCol.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n        tgtCol.semType = DG.SEMTYPE.MACROMOLECULE;\n        return tgtCol;\n    });\n}\nexport function testMSAEnoughMemory(col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const sequencesCount = col.length;\n        const delta = sequencesCount / 100;\n        for (let i = delta; i < sequencesCount; i += delta) {\n            try {\n                yield runKalign(DG.Column.fromStrings(col.name, col.toList().slice(0, Math.round(i))));\n                console.log(`runKalign succeeded on ${i}`);\n            }\n            catch (error) {\n                console.log(`runKalign failed on ${i} with '${error}'`);\n            }\n        }\n    });\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { AvailableMetrics } from '@datagrok-libraries/ml/src/typed-metrics';\nimport * as grok from 'datagrok-api/grok';\nexport function getDistances(col, seq) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const stringArray = col.toList();\n        const distances = new Array(stringArray.length).fill(0);\n        for (let i = 0; i < stringArray.length; ++i) {\n            const distance = stringArray[i] ? AvailableMetrics['String']['Levenshtein'](stringArray[i], seq) : null;\n            distances[i] = distance ? distance / Math.max(stringArray[i].length, seq.length) : null;\n        }\n        return distances;\n    });\n}\nexport function getSimilaritiesMarix(dim, seqCol, df, colName, simArr) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const distances = new Array(simArr.length).fill(null);\n        for (let i = 0; i != dim - 1; ++i) {\n            const seq = seqCol.get(i);\n            df.rows.removeAt(0, 1, false);\n            distances[i] = (yield getDistances(df.col(colName), seq));\n        }\n        for (let i = 0; i < distances.length; i++) {\n            for (let j = 0; j < distances[i].length; j++) {\n                distances[i][j] = getSimilarityFromDistance(distances[i][j]);\n            }\n            simArr[i] = DG.Column.fromList(DG.COLUMN_TYPE.FLOAT, 'distances', distances[i]);\n        }\n        return simArr;\n    });\n}\nexport function createTooltipElement(params) {\n    const tooltipElement = ui.divH([]);\n    const columnNames = ui.divV([\n        ui.divText(params.seqCol.name),\n        ui.divText(params.activityCol.name),\n    ]);\n    columnNames.style.fontWeight = 'bold';\n    columnNames.style.display = 'flex';\n    columnNames.style.justifyContent = 'space-between';\n    tooltipElement.append(columnNames);\n    params.line.mols.forEach((molIdx, idx) => {\n        const activity = ui.divText(params.activityCol.get(molIdx).toFixed(2));\n        activity.style.display = 'flex';\n        activity.style.justifyContent = 'left';\n        activity.style.paddingLeft = '30px';\n        tooltipElement.append(ui.divV([\n            ui.divText(params.seqCol.get(molIdx)),\n            activity,\n        ]));\n    });\n    return tooltipElement;\n}\nfunction moleculeInfo(df, idx, seqColName) {\n    let dict = {};\n    for (let col of df.columns) {\n        if (col.name !== seqColName) {\n            dict[col.name] = df.get(col.name, idx);\n        }\n    }\n    return ui.tableFromMap(dict);\n}\nexport function createPropPanelElement(params) {\n    const propPanel = ui.divV([]);\n    const columnNames = ui.divH([\n        ui.divText(params.seqCol.name),\n        ui.divText(params.activityCol.name),\n    ]);\n    columnNames.style.fontWeight = 'bold';\n    columnNames.style.justifyContent = 'space-between';\n    propPanel.append(columnNames);\n    const hosts = [];\n    params.line.mols.forEach((molIdx, hostIdx) => {\n        const activity = ui.divText(params.activityCol.get(molIdx).toFixed(2));\n        activity.style.paddingLeft = '15px';\n        activity.style.paddingLeft = '10px';\n        const molHost = ui.divText(params.seqCol.get(molIdx));\n        if (params.df.currentRowIdx === molIdx) {\n            molHost.style.border = 'solid 1px lightgrey';\n        }\n        //@ts-ignore\n        ui.tooltip.bind(molHost, () => moleculeInfo(params.df, molIdx, params.seqCol.name));\n        molHost.onclick = () => {\n            const obj = grok.shell.o;\n            molHost.style.border = 'solid 1px lightgrey';\n            params.df.currentRowIdx = molIdx;\n            hosts.forEach((h, i) => {\n                if (i !== hostIdx) {\n                    h.style.border = '';\n                }\n            });\n            setTimeout(() => {\n                grok.shell.o = obj;\n            }, 1000);\n        };\n        propPanel.append(ui.divH([\n            molHost,\n            activity,\n        ]));\n        hosts.push(molHost);\n    });\n    propPanel.append(ui.divH([\n        ui.divText(`Cliff: `, { style: { fontWeight: 'bold', paddingRight: '5px' } }),\n        ui.divText(params.sali.toFixed(2))\n    ]));\n    return propPanel;\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport { reduceDimensinalityWithNormalization } from '@datagrok-libraries/ml/src/sequence-space';\nexport function sequenceSpace(spaceParams) {\n    return __awaiter(this, void 0, void 0, function* () {\n        // code deprecated since seqCol is encoded\n        /*    let preparedData: any;\n          if (!(spaceParams.seqCol!.tags[DG.TAGS.UNITS] === 'HELM')) {\n            const sep = spaceParams.seqCol.getTag(UnitsHandler.TAGS.separator);\n            const sepFinal = sep ? sep === '.' ? '\\\\\\.' : sep : '-';\n            const regex = new RegExp(sepFinal, 'g');\n            if (Object.keys(AvailableMetrics['String']).includes(spaceParams.similarityMetric))\n              preparedData = spaceParams.seqCol.toList().map((v: string) => v.replace(regex, '')) as string[];\n            else\n              preparedData = spaceParams.seqCol.toList().map((v: string) => v.replace(regex, '')) as string[];\n          } else {\n            preparedData = spaceParams.seqCol.toList();\n          }  */\n        const sequenceSpaceResult = yield reduceDimensinalityWithNormalization(spaceParams.seqCol.toList(), spaceParams.methodName, spaceParams.similarityMetric, spaceParams.options);\n        const cols = spaceParams.embedAxesNames.map((name, index) => DG.Column.fromFloat32Array(name, sequenceSpaceResult.embedding[index]));\n        return { distance: sequenceSpaceResult.distance, coordinates: new DG.ColumnList(cols) };\n    });\n}\nexport function getEmbeddingColsNames(df) {\n    const axes = ['Embed_X', 'Embed_Y'];\n    const colNameInd = df.columns.names().filter((it) => it.includes(axes[0])).length + 1;\n    return axes.map((it) => `${it}_${colNameInd}`);\n}\n","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport * as grok from 'datagrok-api/grok';\nimport { CAP_GROUP_NAME, CAP_GROUP_SMILES, jsonSdfMonomerLibDict, MONOMER_ENCODE_MAX, MONOMER_ENCODE_MIN, MONOMER_SYMBOL, RGROUP_ALTER_ID, RGROUP_FIELD, RGROUP_LABEL, SDF_MONOMER_NAME } from '../const';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nexport const HELM_CORE_LIB_FILENAME = '/samples/HELMCoreLibrary.json';\nexport const HELM_CORE_LIB_MONOMER_SYMBOL = 'symbol';\nexport const HELM_CORE_LIB_MOLFILE = 'molfile';\nexport const HELM_CORE_FIELDS = ['symbol', 'molfile', 'rgroups', 'name'];\nexport function encodeMonomers(col) {\n    let encodeSymbol = MONOMER_ENCODE_MIN;\n    const monomerSymbolDict = {};\n    const units = col.tags[DG.TAGS.UNITS];\n    const sep = col.getTag(UnitsHandler.TAGS.separator);\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const encodedStringArray = [];\n    for (let i = 0; i < col.length; ++i) {\n        let encodedMonomerStr = '';\n        const monomers = splitterFunc(col.get(i));\n        monomers.forEach((m) => {\n            if (!monomerSymbolDict[m]) {\n                if (encodeSymbol > MONOMER_ENCODE_MAX) {\n                    grok.shell.error(`Not enough symbols to encode monomers`);\n                    return null;\n                }\n                monomerSymbolDict[m] = encodeSymbol;\n                encodeSymbol++;\n            }\n            encodedMonomerStr += String.fromCodePoint(monomerSymbolDict[m]);\n        });\n        encodedStringArray.push(encodedMonomerStr);\n    }\n    return DG.Column.fromStrings('encodedMolecules', encodedStringArray);\n}\nexport function getMolfilesFromSeq(col, monomersLibObject) {\n    const units = col.tags[DG.TAGS.UNITS];\n    const sep = col.getTag('separator');\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const monomersDict = createMomomersMolDict(monomersLibObject);\n    const molFiles = [];\n    for (let i = 0; i < col.length; ++i) {\n        const macroMolecule = col.get(i);\n        const monomers = splitterFunc(macroMolecule);\n        const molFilesForSeq = [];\n        for (let j = 0; j < monomers.length; ++j) {\n            if (monomers[j]) {\n                if (!monomersDict[monomers[j]]) {\n                    grok.shell.warning(`Monomer ${monomers[j]} is missing in HELM library. Structure cannot be created`);\n                    return null;\n                }\n                molFilesForSeq.push(JSON.parse(JSON.stringify(monomersDict[monomers[j]])));\n            }\n        }\n        molFiles.push(molFilesForSeq);\n    }\n    return molFiles;\n}\nexport function getMolfilesFromSingleSeq(cell, monomersLibObject) {\n    const units = cell.column.tags[DG.TAGS.UNITS];\n    const sep = cell.column.getTag('separator');\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const monomersDict = createMomomersMolDict(monomersLibObject);\n    const molFiles = [];\n    const macroMolecule = cell.value;\n    const monomers = splitterFunc(macroMolecule);\n    const molFilesForSeq = [];\n    for (let j = 0; j < monomers.length; ++j) {\n        if (monomers[j]) {\n            if (!monomersDict[monomers[j]]) {\n                grok.shell.warning(`Monomer ${monomers[j]} is missing in HELM library. Structure cannot be created`);\n                return null;\n            }\n            molFilesForSeq.push(JSON.parse(JSON.stringify(monomersDict[monomers[j]])));\n        }\n    }\n    molFiles.push(molFilesForSeq);\n    return molFiles;\n}\nexport function createMomomersMolDict(lib) {\n    const dict = {};\n    lib.forEach((it) => {\n        if (it['polymerType'] === 'PEPTIDE') {\n            const monomerObject = {};\n            HELM_CORE_FIELDS.forEach((field) => {\n                monomerObject[field] = it[field];\n            });\n            dict[it[HELM_CORE_LIB_MONOMER_SYMBOL]] = monomerObject;\n        }\n    });\n    return dict;\n}\nexport function createJsonMonomerLibFromSdf(table) {\n    const resultLib = [];\n    for (let i = 0; i < table.rowCount; i++) {\n        const monomer = {};\n        Object.keys(jsonSdfMonomerLibDict).forEach((key) => {\n            if (key === MONOMER_SYMBOL) {\n                const monomerSymbol = table.get(jsonSdfMonomerLibDict[key], i);\n                monomer[key] = monomerSymbol === '.' ? table.get(SDF_MONOMER_NAME, i) : monomerSymbol;\n            }\n            else if (key === RGROUP_FIELD) {\n                const rgroups = table.get(jsonSdfMonomerLibDict[key], i).split('\\n');\n                const jsonRgroups = [];\n                rgroups.forEach((g) => {\n                    const rgroup = {};\n                    const altAtom = g.substring(g.lastIndexOf(']') + 1);\n                    const radicalNum = g.match(/\\[R(\\d+)\\]/)[1];\n                    rgroup[CAP_GROUP_SMILES] = altAtom === 'H' ? `[*:${radicalNum}][H]` : `O[*:${radicalNum}]`;\n                    rgroup[RGROUP_ALTER_ID] = altAtom === 'H' ? `R${radicalNum}-H` : `R${radicalNum}-OH`;\n                    rgroup[CAP_GROUP_NAME] = altAtom === 'H' ? `H` : `OH`;\n                    rgroup[RGROUP_LABEL] = `R${radicalNum}`;\n                    jsonRgroups.push(rgroup);\n                });\n                monomer[key] = jsonRgroups;\n            }\n            else {\n                if (jsonSdfMonomerLibDict[key])\n                    monomer[key] = table.get(jsonSdfMonomerLibDict[key], i);\n            }\n        });\n        resultLib.push(monomer);\n    }\n    return resultLib;\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport { VdRegionType } from '@datagrok-libraries/bio/src/vd-regions';\nconst vrt = VdRegionType;\n// Positions of regions for numbering schemes\n// http://www.bioinf.org.uk/abs/info.html\n// const imgtRegions: VdRegion[] = [\n//   new VdRegion(vrt.FR, 'FR1', 'Light', 1, '1', '26'),\n//   new VdRegion(vrt.FR, 'FR1', 'Heavy', 1, '1', '26'),\n//\n//   new VdRegion(vrt.CDR, 'CDR1', 'Light', 2, '27', '38'), // 27-32\n//   new VdRegion(vrt.CDR, 'CDR1', 'Heavy', 2, '27', '38'), // 27-32\n//\n//   new VdRegion(vrt.FR, 'FR2', 'Light', 3, '39', '55'),\n//   new VdRegion(vrt.FR, 'FR2', 'Heavy', 3, '39', '55'),\n//\n//   new VdRegion(vrt.CDR, 'CDR2', 'Light', 4, '56', '65'),\n//   new VdRegion(vrt.CDR, 'CDR2', 'Heavy', 4, '56', '65'),\n//\n//   new VdRegion(vrt.FR, 'FR3', 'Light', 5, '66', '104'),\n//   new VdRegion(vrt.FR, 'FR3', 'Heavy', 5, '66', '104'),\n//\n//   new VdRegion(vrt.CDR, 'CDR3', 'Light', 6, '105', '117'),\n//   new VdRegion(vrt.CDR, 'CDR3', 'Heavy', 6, '105', '117'),\n//\n//   new VdRegion(vrt.FR, 'FR4', 'Light', 7, '118', null/*127*/),\n//   new VdRegion(vrt.FR, 'FR4', 'Heavy', 7, '118', null/*128*/),\n// ];\n/** Viewer with tabs based on description of chain regions.\n *  Used to define regions of an immunoglobulin LC.\n */\nexport class VdRegionsViewer extends DG.JsViewer {\n    constructor() {\n        super();\n        // private regionsDf: DG.DataFrame;\n        this.regionsFg = null;\n        // private regionsTV: DG.TableView;\n        this.regionsRoot = null;\n        this.isOpened = false;\n        this.panelNode = null;\n        this.regions = [];\n        // #region -- Handle controls' events --\n        this.resizing = false;\n        // #endregion\n        //#region -- View --\n        this.host = null;\n        this.mainLayout = null;\n        this.logos = [];\n        // To prevent ambiguous numbering scheme in MLB\n        this.regionTypes = this.stringList('regionTypes', [vrt.CDR], { choices: Object.values(vrt).filter((t) => t != vrt.Unknown) });\n        this.chains = this.stringList('chains', ['Heavy', 'Light'], { choices: ['Heavy', 'Light'] });\n        this.sequenceColumnNamePostfix = this.string('sequenceColumnNamePostfix', 'chain sequence');\n        this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n        this.positionWidth = this.float('positionWidth', 16);\n    }\n    get df() {\n        return this.dataFrame;\n    }\n    // TODO: .onTableAttached is not calling on dataFrame set, onPropertyChanged  also not calling\n    setDf(value, regions) {\n        return __awaiter(this, void 0, void 0, function* () {\n            console.debug('VdRegionsViewer.setDf()');\n            yield this.destroyView();\n            this.regions = regions;\n            this.dataFrame = value;\n            yield this.buildView();\n        });\n    }\n    init() {\n        return __awaiter(this, void 0, void 0, function* () {\n            //#region regionsDF with filter\n            // this.regionsDf = DG.DataFrame.fromObjects(this.regions);\n            // this.regionsDf.rows.filter((row) => row.name == 'CDR1');\n            // // To available options /\n            // this.regionsFg = (await this.regionsDf.plot.fromType(DG.VIEWER.FILTERS, {\n            //   // columnNames: ['name',],\n            //   showFilterCountsIndication: false,\n            //   showHeader: false,\n            //   showSearchBox: false,\n            //   filters: [\n            //     {type: DG.FILTER_TYPE.CATEGORICAL, column: 'type', label: 'Region name', showHistogram: false},\n            //     {type: DG.FILTER_TYPE.CATEGORICAL, column: 'name', label: 'Region type', showHistogram: false},\n            //   ],\n            //   title: 'Regions filter',\n            //   showTitle: true,\n            //   description: 'Filter for regions of multiple alignment by IMGT nomenclature',\n            //   someProperty: 'Hello',\n            // })) as DG.FilterGroup;\n            //#endregion regionsDF with filter\n            // this.mlbView.dockManager.dock(this.regionsFg.root, DG.DOCK_TYPE.LEFT, rootNode, 'Filter regions', 0.2);\n            this.subs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n            // rxjs.fromEvent(this.root, 'mousemove').subscribe(this.onMouseMoveRoot.bind(this));\n            this.root.addEventListener('mousemove', this.onMouseMoveRoot.bind(this));\n            yield this.buildView();\n        });\n    }\n    onTableAttached() {\n        return __awaiter(this, void 0, void 0, function* () {\n            yield this.init();\n        });\n    }\n    onPropertyChanged(property) {\n        const _super = Object.create(null, {\n            onPropertyChanged: { get: () => super.onPropertyChanged }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            _super.onPropertyChanged.call(this, property);\n            if (property) {\n                switch (property.name) {\n                    case 'regionTypes':\n                        break;\n                    case 'chains':\n                        break;\n                    case 'sequenceColumnNamePostfix':\n                        break;\n                    case 'skipEmptyPositions':\n                        // for (let orderI = 0; orderI < this.logos.length; orderI++) {\n                        //   for (let chainI = 0; chainI < this.chains.length; chainI++) {\n                        //     const chain: string = this.chains[chainI];\n                        //     this.logos[orderI][chain].setOptions({skipEmptyPositions: this.skipEmptyPositions});\n                        //   }\n                        // }\n                        // this.calcSize();\n                        yield this.destroyView();\n                        yield this.buildView();\n                        break;\n                    case 'positionWidth':\n                        yield this.destroyView();\n                        yield this.buildView();\n                        break;\n                }\n            }\n        });\n    }\n    reset() {\n        return __awaiter(this, void 0, void 0, function* () {\n        });\n    }\n    open(mlbView) {\n        return __awaiter(this, void 0, void 0, function* () {\n            if (!this.isOpened) {\n                this.isOpened = true;\n                this.panelNode = mlbView.dockManager.dock(this.root, DG.DOCK_TYPE.TOP, null, 'Regions', 0.2);\n            }\n        });\n    }\n    show(mlbView) {\n        return __awaiter(this, void 0, void 0, function* () {\n        });\n    }\n    rootOnSizeChanged(args) {\n        this.calcSize();\n    }\n    destroyView() {\n        return __awaiter(this, void 0, void 0, function* () {\n            // TODO: Unsubscribe from and remove all view elements\n            console.debug(`VdRegionsViewer.destroyView( mainLayout = ${!this.mainLayout ? 'none' : 'value'} )`);\n            if (this.mainLayout != null) {\n                // this.root.removeChild(this.host);\n                this.mainLayout.remove();\n                this.host.remove();\n                this.host = null;\n                this.mainLayout = null;\n            }\n        });\n    }\n    buildView() {\n        return __awaiter(this, void 0, void 0, function* () {\n            console.debug('VdRegionsViewer.buildView() start');\n            const colNames = Object.assign({}, ...this.chains.map((chain) => ({ [chain]: `${chain} ${this.sequenceColumnNamePostfix}` })));\n            const regionsFiltered = this.regions.filter((r) => this.regionTypes.includes(r.type));\n            const orderList = Array.from(new Set(regionsFiltered.map((r) => r.order))).sort();\n            this.logos = [];\n            for (let orderI = 0; orderI < orderList.length; orderI++) {\n                const regionChains = {};\n                for (const chain of this.chains) {\n                    const region = regionsFiltered\n                        .find((r) => r.order == orderList[orderI] && r.chain == chain);\n                    regionChains[chain] = (yield this.dataFrame.plot.fromType('WebLogo', {\n                        sequenceColumnName: colNames[chain],\n                        startPositionName: region.positionStartName,\n                        endPositionName: region.positionEndName,\n                        fixWidth: true,\n                        skipEmptyPositions: this.skipEmptyPositions,\n                        positionWidth: this.positionWidth,\n                    }));\n                }\n                // WebLogo creation fires onRootSizeChanged event even before control being added to this.logos\n                this.logos[orderI] = regionChains;\n            }\n            // ui.tableFromMap()\n            // DG.HtmlTable.create()\n            this.mainLayout = ui.table(this.chains, (chain) => {\n                const elements = [\n                    // This is chain label\n                    ...(orderList.length > 0 ? [ui.div(chain, {\n                            style: {\n                                transform: 'rotate(-90deg)',\n                                font: '12px Roboto, Roboto Local, sans-serif',\n                                textAlign: 'center',\n                                width: '16px',\n                                marginTop: '24px',\n                                marginLeft: '6px',\n                            }\n                        })] : []),\n                    // List with controls for regions\n                    ...[...Array(orderList.length).keys()].map((orderI) => {\n                        const wl = this.logos[orderI][chain];\n                        wl.root.style.height = '100%';\n                        const resDiv = ui.div([wl.root] /*`${chain} ${regionsFiltered[rI]}`*/, {\n                            style: {\n                                // height: '100%',\n                                marginTop: '4px',\n                                marginBottom: '4px',\n                            }\n                        });\n                        return resDiv;\n                    })\n                ];\n                return elements;\n            }, ['', ...[...Array(orderList.length).keys()].map((orderI) => regionsFiltered.find((r) => r.order == orderList[orderI] && r.chain == this.chains[0]).name || 'Name')]);\n            this.mainLayout.className = 'mlb-vd-regions-viewer-table2';\n            // this.mainLayout.style.background = '#EEEEFF';\n            // this.mainLayout.style.height = '100%';\n            // this.mainLayout.style.border = '1px solid black';\n            const color = `#ffbb${Math.ceil(Math.random() * 255).toString(16)}`;\n            this.host = ui.box(this.mainLayout, { /*style: {backgroundColor: color}*/});\n            this.root.appendChild(this.host);\n            this.root.style.overflowX = 'auto';\n            this.calcSize();\n            console.debug('VdRegionsViewer.buildView() end');\n        });\n    }\n    calcSize() {\n        const logoHeight = (this.root.clientHeight - 54) / this.chains.length;\n        const maxHeight = Math.min(logoHeight, Math.max(...this.logos.map((wlDict) => Math.max(...Object.values(wlDict).map((wl) => wl.maxHeight)))));\n        for (let orderI = 0; orderI < this.logos.length; orderI++) {\n            for (let chainI = 0; chainI < this.chains.length; chainI++) {\n                const chain = this.chains[chainI];\n                this.logos[orderI][chain].root.style.height = `${maxHeight}px`;\n            }\n        }\n    }\n    //#endregion -- View --\n    onMouseMoveRoot(e) {\n        // ui.tooltip.show('text', e.x + 8, e.y + 8,);\n        // console.log(`onMouseMoveRoot.( x: ${e.x}, y: ${e.y} )`);\n    }\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport { getMolfilesFromSingleSeq } from '../utils/utils';\nimport { getMacroMol } from '../utils/atomic-works';\n/**\n * 3D representation widget of macromolecule.\n *\n * @export\n * @param {DG.Cell} macroMolecule macromolecule cell.\n * @param {any[]} monomersLibObject\n * @return {Promise<DG.Widget>} Widget.\n */\nexport function representationsWidget(macroMolecule, monomersLibObject) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const pi = DG.TaskBarProgressIndicator.create('Creating 3D view');\n        let widgetHost;\n        let molBlock3D = '';\n        try {\n            try {\n                const atomicCodes = getMolfilesFromSingleSeq(macroMolecule, monomersLibObject);\n                const result = yield getMacroMol(atomicCodes);\n                const molBlock2D = result[0];\n                molBlock3D = (yield grok.functions.call('Bio:Embed', { molBlock2D }));\n            }\n            catch (e) {\n                console.warn(e);\n            }\n            try {\n                molBlock3D = molBlock3D.replaceAll('\\\\n', '\\n');\n                const stringBlob = new Blob([molBlock3D], { type: 'text/plain' });\n                const nglHost = ui.div([], { classes: 'd4-ngl-viewer', id: 'ngl-3d-host' });\n                //@ts-ignore\n                const stage = new NGL.Stage(nglHost, { backgroundColor: 'white' });\n                //@ts-ignore\n                stage.loadFile(stringBlob, { ext: 'sdf' }).then(function (comp) {\n                    stage.setSize(300, 300);\n                    comp.addRepresentation('ball+stick');\n                    comp.autoView();\n                });\n                const sketch = grok.chem.svgMol(molBlock3D);\n                const panel = ui.divH([sketch]);\n                widgetHost = ui.div([panel, nglHost]);\n            }\n            catch (e) {\n                widgetHost = ui.divText('Couldn\\'t get peptide structure');\n            }\n        }\n        catch (e) {\n            widgetHost = ui.divText('Couldn\\'t get peptide structure');\n        }\n        pi.close();\n        return new DG.Widget(widgetHost);\n    });\n}\n","module.exports = $;","module.exports = DG;","module.exports = grok;","module.exports = rxjs;","module.exports = ui;","module.exports = wu;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","// This function allow to reference async chunks\n__webpack_require__.u = (chunkId) => {\n\t// return url for filenames based on template\n\treturn \"\" + chunkId + \".js\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","var scriptUrl;\nif (__webpack_require__.g.importScripts) scriptUrl = __webpack_require__.g.location + \"\";\nvar document = __webpack_require__.g.document;\nif (!scriptUrl && document) {\n\tif (document.currentScript)\n\t\tscriptUrl = document.currentScript.src\n\tif (!scriptUrl) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tif(scripts.length) scriptUrl = scripts[scripts.length - 1].src\n\t}\n}\n// When supporting browsers where an automatic publicPath is not supported you must specify an output.publicPath manually via configuration\n// or pass an empty string (\"\") and set the __webpack_public_path__ variable from your code to use your own logic.\nif (!scriptUrl) throw new Error(\"Automatic publicPath is not supported in this browser\");\nscriptUrl = scriptUrl.replace(/#.*$/, \"\").replace(/\\?.*$/, \"\").replace(/\\/[^\\/]+$/, \"/\");\n__webpack_require__.p = scriptUrl;","__webpack_require__.b = document.baseURI || self.location.href;\n\n// object to store loaded and loading chunks\n// undefined = chunk not loaded, null = chunk preloaded/prefetched\n// [resolve, reject, Promise] = chunk loading, 0 = chunk loaded\nvar installedChunks = {\n\t\"package\": 0\n};\n\n// no chunk on demand loading\n\n// no prefetching\n\n// no preloaded\n\n// no HMR\n\n// no HMR manifest\n\n// no on chunks loaded\n\n// no jsonp function","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\n/* Do not change these import lines to match external modules in webpack configuration */\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nexport const _package = new DG.Package();\nimport { MacromoleculeDifferenceCellRenderer, MonomerCellRenderer } from './utils/cell-renderer';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport { VdRegionsViewer } from './viewers/vd-regions-viewer';\nimport { runKalign } from './utils/multiple-sequence-alignment';\nimport { SequenceAlignment } from './seq_align';\nimport { getEmbeddingColsNames, sequenceSpace } from './utils/sequence-space';\nimport { getActivityCliffs } from '@datagrok-libraries/ml/src/viewers/activity-cliffs';\nimport { createPropPanelElement, createTooltipElement, getSimilaritiesMarix } from './utils/sequence-activity-cliffs';\nimport { createJsonMonomerLibFromSdf, encodeMonomers, getMolfilesFromSeq, HELM_CORE_LIB_FILENAME } from './utils/utils';\nimport { getMacroMol } from './utils/atomic-works';\nimport { MacromoleculeSequenceCellRenderer } from './utils/cell-renderer';\nimport { convert } from './utils/convert';\nimport { representationsWidget } from './widgets/representations';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nimport { FastaFileHandler } from '@datagrok-libraries/bio/src/utils/fasta-handler';\nimport { removeEmptyStringRows } from '@datagrok-libraries/utils/src/dataframe-utils';\nimport { splitAlignedSequences } from '@datagrok-libraries/bio/src/utils/splitter';\nimport * as C from './utils/constants';\n//tags: init\nexport function initBio() {\n    return __awaiter(this, void 0, void 0, function* () {\n    });\n}\n//name: fastaSequenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: Sequence\n//meta.columnTags: units=fasta\n//output: grid_cell_renderer result\nexport function fastaSequenceCellRenderer() {\n    return new MacromoleculeSequenceCellRenderer();\n}\n//name: separatorSequenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: Sequence\n//meta.columnTags: units=separator\n//output: grid_cell_renderer result\nexport function separatorSequenceCellRenderer() {\n    return new MacromoleculeSequenceCellRenderer();\n}\nfunction checkInputColumnUi(col, name, allowedNotations = [], allowedAlphabets = []) {\n    const [res, msg] = checkInputColumn(col, name, allowedNotations, allowedAlphabets);\n    if (!res)\n        grok.shell.warning(msg);\n    return res;\n}\nexport function checkInputColumn(col, name, allowedNotations = [], allowedAlphabets = []) {\n    let res = true;\n    let msg = '';\n    const uh = new UnitsHandler(col);\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE) {\n        grok.shell.warning(name + ' analysis is allowed for Macromolecules semantic type');\n        res = false;\n    }\n    else {\n        const notation = uh.notation;\n        if (allowedNotations.length > 0 &&\n            !allowedNotations.some((n) => notation.toUpperCase() == (n.toUpperCase()))) {\n            const notationAdd = allowedNotations.length == 0 ? 'any notation' :\n                (`notation${allowedNotations.length > 1 ? 's' : ''} ${allowedNotations.map((n) => `\"${n}\"`).join(', ')} `);\n            msg = `${name} + ' analysis is allowed for Macromolecules with notation ${notationAdd}.`;\n            res = false;\n        }\n        else if (!uh.isHelm()) {\n            // alphabet is not specified for 'helm' notation\n            const alphabet = uh.alphabet;\n            if (allowedAlphabets.length > 0 &&\n                !allowedAlphabets.some((a) => alphabet.toUpperCase() == (a.toUpperCase()))) {\n                const alphabetAdd = allowedAlphabets.length == 0 ? 'any alphabet' :\n                    (`alphabet${allowedAlphabets.length > 1 ? 's' : ''} ${allowedAlphabets.map((a) => `\"${a}\"`).join(', ')}.`);\n                msg = `${name} + ' analysis is allowed for Macromolecules with alphabet ${alphabetAdd}.`;\n                res = false;\n            }\n        }\n    }\n    return [res, msg];\n}\n//name: sequenceAlignment\n//input: string alignType {choices: ['Local alignment', 'Global alignment']}\n// eslint-disable-next-line max-len\n//input: string alignTable {choices: ['AUTO', 'NUCLEOTIDES', 'BLOSUM45', 'BLOSUM50','BLOSUM62','BLOSUM80','BLOSUM90','PAM30','PAM70','PAM250','SCHNEIDER','TRANS']}\n//input: double gap\n//input: string seq1\n//input: string seq2\n//output: object res\nexport function sequenceAlignment(alignType, alignTable, gap, seq1, seq2) {\n    const toAlign = new SequenceAlignment(seq1, seq2, gap, alignTable);\n    const res = alignType == 'Local alignment' ? toAlign.smithWaterman() : toAlign.needlemanWunch();\n    return res;\n}\n//name: WebLogo\n//description: WebLogo viewer\n//tags: viewer, panel\n//output: viewer result\nexport function webLogoViewer() {\n    return new WebLogo();\n}\n//name: VdRegions\n//description: V-Domain regions viewer\n//tags: viewer, panel\n//output: viewer result\nexport function vdRegionViewer() {\n    return new VdRegionsViewer();\n}\n//top-menu: Bio | Sequence Activity Cliffs...\n//name: Sequence Activity Cliffs\n//description: detect activity cliffs\n//input: dataframe table [Input data table]\n//input: column macroMolecule {semType: Macromolecule}\n//input: column activities\n//input: double similarity = 90 [Similarity cutoff]\n//input: string methodName { choices:[\"UMAP\", \"t-SNE\", \"SPE\"] }\nexport function activityCliffs(df, macroMolecule, activities, similarity, methodName) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!checkInputColumnUi(macroMolecule, 'Activity Cliffs'))\n            return;\n        const encodedCol = encodeMonomers(macroMolecule);\n        if (!encodedCol)\n            return;\n        const axesNames = getEmbeddingColsNames(df);\n        const options = {\n            'SPE': { cycles: 2000, lambda: 1.0, dlambda: 0.0005 },\n        };\n        const tags = {\n            'units': macroMolecule.tags['units'],\n            'aligned': macroMolecule.tags['aligned'],\n            'separator': macroMolecule.tags['separator'],\n            'alphabet': macroMolecule.tags['alphabet'],\n        };\n        const sp = yield getActivityCliffs(df, macroMolecule, encodedCol, axesNames, 'Activity cliffs', activities, similarity, 'Levenshtein', methodName, DG.SEMTYPE.MACROMOLECULE, tags, sequenceSpace, getSimilaritiesMarix, createTooltipElement, createPropPanelElement, options[methodName]);\n        return sp;\n    });\n}\n//top-menu: Bio | Sequence Space...\n//name: Sequence Space\n//input: dataframe table\n//input: column macroMolecule { semType: Macromolecule }\n//input: string methodName { choices:[\"UMAP\", \"t-SNE\", \"SPE\"] }\n//input: string similarityMetric { choices:[\"Levenshtein\", \"Tanimoto\"] }\n//input: bool plotEmbeddings = true\nexport function sequenceSpaceTopMenu(table, macroMolecule, methodName, similarityMetric = 'Levenshtein', plotEmbeddings) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!checkInputColumnUi(macroMolecule, 'Activity Cliffs'))\n            return;\n        const encodedCol = encodeMonomers(macroMolecule);\n        if (!encodedCol)\n            return;\n        const embedColsNames = getEmbeddingColsNames(table);\n        const withoutEmptyValues = DG.DataFrame.fromColumns([encodedCol]).clone();\n        const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, encodedCol);\n        const chemSpaceParams = {\n            seqCol: withoutEmptyValues.col(encodedCol.name),\n            methodName: methodName,\n            similarityMetric: similarityMetric,\n            embedAxesNames: embedColsNames\n        };\n        const sequenceSpaceRes = yield sequenceSpace(chemSpaceParams);\n        const embeddings = sequenceSpaceRes.coordinates;\n        for (const col of embeddings) {\n            const listValues = col.toList();\n            emptyValsIdxs.forEach((ind) => listValues.splice(ind, 0, null));\n            table.columns.add(DG.Column.fromList('double', col.name, listValues));\n        }\n        let sp;\n        if (plotEmbeddings) {\n            for (const v of grok.shell.views) {\n                if (v.name === table.name)\n                    sp = v.scatterPlot({ x: embedColsNames[0], y: embedColsNames[1], title: 'Sequence space' });\n            }\n        }\n        return sp;\n    });\n}\n;\n//top-menu: Bio | To Atomic Level...\n//name: To Atomic Level\n//description: returns molfiles for each monomer from HELM library\n//input: dataframe df [Input data table]\n//input: column macroMolecule {semType: Macromolecule}\nexport function toAtomicLevel(df, macroMolecule) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (DG.Func.find({ package: 'Chem', name: 'getRdKitModule' }).length === 0) {\n            grok.shell.warning('Transformation to atomic level requires package \"Chem\" installed.');\n            return;\n        }\n        if (!checkInputColumnUi(macroMolecule, 'To Atomic Level'))\n            return;\n        const monomersLibFile = yield _package.files.readAsText(HELM_CORE_LIB_FILENAME);\n        const monomersLibObject = JSON.parse(monomersLibFile);\n        const atomicCodes = getMolfilesFromSeq(macroMolecule, monomersLibObject);\n        const result = yield getMacroMol(atomicCodes);\n        const col = DG.Column.fromStrings('regenerated', result);\n        col.semType = DG.SEMTYPE.MOLECULE;\n        col.tags[DG.TAGS.UNITS] = 'molblock';\n        df.columns.add(col, true);\n        yield grok.data.detectSemanticTypes(df);\n    });\n}\n//top-menu: Bio | MSA...\n//name: MSA\n//input: dataframe table\n//input: column sequence { semType: Macromolecule, units: ['fasta'], alphabet: ['DNA', 'RNA', 'PT'] }\n//output: column result\nexport function multipleSequenceAlignmentAny(table, col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const func = DG.Func.find({ package: 'Bio', name: 'multipleSequenceAlignmentAny' })[0];\n        if (!checkInputColumnUi(col, 'MSA', ['fasta'], ['DNA', 'RNA', 'PT']))\n            return null;\n        const unUsedName = table.columns.getUnusedName(`msa(${col.name})`);\n        const msaCol = yield runKalign(col, false, unUsedName);\n        table.columns.add(msaCol);\n        // This call is required to enable cell renderer activation\n        yield grok.data.detectSemanticTypes(table);\n        // const tv: DG.TableView = grok.shell.tv;\n        // tv.grid.invalidate();\n        return msaCol;\n    });\n}\n//name: Bio | MSA\n//tags: bio, panel\n//input: column sequence { semType: Macromolecule }\n//output: column result\nexport function panelMSA(col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        return multipleSequenceAlignmentAny(col.dataFrame, col);\n    });\n}\n//name: Composition Analysis\n//top-menu: Bio | Composition Analysis\n//output: viewer result\nexport function compositionAnalysis() {\n    return __awaiter(this, void 0, void 0, function* () {\n        // Higher priority for columns with MSA data to show with WebLogo.\n        const tv = grok.shell.tv;\n        const df = tv.dataFrame;\n        //@ts-ignore\n        const colList = df.columns.toList().filter((col) => {\n            if (col.semType != DG.SEMTYPE.MACROMOLECULE)\n                return false;\n            const colUH = new UnitsHandler(col);\n            // TODO: prevent for cyclic, branched or multiple chains in Helm\n            return true;\n        });\n        const handler = (col) => __awaiter(this, void 0, void 0, function* () {\n            if (!checkInputColumnUi(col, 'Composition'))\n                return;\n            const wlViewer = tv.addViewer('WebLogo', { sequenceColumnName: col.name });\n            grok.shell.tv.dockManager.dock(wlViewer, DG.DOCK_TYPE.DOWN, null, 'Composition analysis', 0.25);\n        });\n        let col = null;\n        if (colList.length == 0) {\n            grok.shell.error('Current table does not contain sequences');\n            return;\n        }\n        else if (colList.length > 1) {\n            const colListNames = colList.map((col) => col.name);\n            const colInput = ui.choiceInput('Column', colListNames[0], colListNames);\n            ui.dialog({\n                title: 'R-Groups Analysis',\n                helpUrl: '/help/domains/bio/macromolecules.md#composition-analysis'\n            })\n                .add(ui.div([\n                colInput,\n            ]))\n                .onOK(() => __awaiter(this, void 0, void 0, function* () {\n                var _a;\n                const col = (_a = colList.find((col) => col.name == colInput.value)) !== null && _a !== void 0 ? _a : null;\n                if (col)\n                    yield handler(col);\n            }))\n                .show();\n        }\n        else {\n            col = colList[0];\n        }\n        if (!col)\n            return;\n        yield handler(col);\n    });\n}\n//top-menu: Bio | Sdf to Json lib...\n//name: sdfToJsonLib\n//input: dataframe table\nexport function sdfToJsonLib(table) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const jsonMonomerLibrary = createJsonMonomerLibFromSdf(table);\n    });\n}\n//name: Representations\n//tags: panel, widgets\n//input: cell macroMolecule {semType: Macromolecule}\n//output: widget result\nexport function peptideMolecule(macroMolecule) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const monomersLibFile = yield _package.files.readAsText(HELM_CORE_LIB_FILENAME);\n        const monomersLibObject = JSON.parse(monomersLibFile);\n        return representationsWidget(macroMolecule, monomersLibObject);\n    });\n}\n//name: importFasta\n//description: Opens FASTA file\n//tags: file-handler\n//meta.ext: fasta, fna, ffn, faa, frn, fa, fst\n//input: string fileContent\n//output: list tables\nexport function importFasta(fileContent) {\n    const ffh = new FastaFileHandler(fileContent);\n    return ffh.importFasta();\n}\n//name: Bio | Convert ...\n//friendly-name: Bio | Convert\n//tags: panel, bio\n//input: column col {semType: Macromolecule}\nexport function convertPanel(col) {\n    convert(col);\n}\n//name: monomerCellRenderer\n//tags: cellRenderer\n//meta.cellType: Monomer\n//output: grid_cell_renderer result\nexport function monomerCellRenderer() {\n    return new MonomerCellRenderer();\n}\n//name: MacromoleculeDifferenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: MacromoleculeDifference\n//output: grid_cell_renderer result\nexport function macromoleculeDifferenceCellRenderer() {\n    return new MacromoleculeDifferenceCellRenderer();\n}\n//name: testDetectMacromolecule\n//input: string path {choices: ['Demo:Files/', 'System:AppData/']}\n//output: dataframe result\nexport function testDetectMacromolecule(path) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const pi = DG.TaskBarProgressIndicator.create('Test detectMacromolecule...');\n        const fileList = yield grok.dapi.files.list(path, true, '');\n        //@ts-ignore\n        const fileListToTest = fileList.filter((fi) => fi.fileName.endsWith('.csv'));\n        let readyCount = 0;\n        const res = [];\n        for (const fileInfo of fileListToTest) {\n            try {\n                const csv = yield grok.dapi.files.readAsText(path + fileInfo.fullPath);\n                const df = DG.DataFrame.fromCsv(csv);\n                for (const col of df.columns) {\n                    const semType = yield grok.functions.call('Bio:detectMacromolecule', { col: col });\n                    if (semType === DG.SEMTYPE.MACROMOLECULE) {\n                        //console.warn(`file: ${fileInfo.path}, column: ${col.name}, ` +\n                        //  `semType: ${semType}, units: ${col.getTag(DG.TAGS.UNITS)}`);\n                        // console.warn('file: \"' + fileInfo.path + '\", semType: \"' + semType + '\", ' +\n                        //   'units: \"' + col.getTag(DG.TAGS.UNITS) + '\"');\n                        res.push({\n                            file: fileInfo.path, result: 'detected', column: col.name,\n                            message: `units: ${col.getTag(DG.TAGS.UNITS)}`\n                        });\n                    }\n                }\n            }\n            catch (err) {\n                // console.error('file: ' + fileInfo.path + ', error: ' + ex.toString());\n                res.push({\n                    file: fileInfo.path, result: 'error', column: null,\n                    message: err instanceof Error ? err.message : err.toString(),\n                });\n            }\n            finally {\n                readyCount += 1;\n                pi.update(100 * readyCount / fileListToTest.length, `Test ${fileInfo.fileName}`);\n            }\n        }\n        grok.shell.info('Test Demo:Files for detectMacromolecule finished.');\n        pi.close();\n        const resDf = DG.DataFrame.fromObjects(res);\n        resDf.name = `datasets_detectMacromolecule_${path}`;\n        return resDf;\n    });\n}\n//name: Bio | Split to monomers\n//tags: panel, bio\n//input: column col {semType: Macromolecule}\nexport function splitToMonomers(col) {\n    var _a;\n    if (!col.getTag(UnitsHandler.TAGS.aligned).includes(C.MSA))\n        return grok.shell.error('Splitting is applicable only for aligned sequences');\n    const tempDf = splitAlignedSequences(col);\n    const originalDf = col.dataFrame;\n    for (const tempCol of tempDf.columns) {\n        const newCol = originalDf.columns.add(tempCol);\n        newCol.semType = C.SEM_TYPES.MONOMER;\n        newCol.setTag(DG.TAGS.CELL_RENDERER, C.SEM_TYPES.MONOMER);\n        newCol.setTag(C.TAGS.ALPHABET, col.getTag(C.TAGS.ALPHABET));\n    }\n    (_a = grok.shell.v) === null || _a === void 0 ? void 0 : _a.grid.invalidate();\n}\n//name: Bio: getHelmMonomers\n//input: column col {semType: Macromolecule}\n//output: string[] result\nexport function getHelmMonomers(seqCol) {\n    const stats = WebLogo.getStats(seqCol, 5, WebLogo.splitterAsHelm);\n    return Object.keys(stats.freq);\n}\n"],"names":[],"sourceRoot":""}
|
|
7442
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"package.js","mappings":";;;;;;;;;;AAAA;AACA,EAAE,KAA4D;AAC9D,EAAE,CACmD;AACrD,CAAC,sBAAsB;;AAEvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,eAAe;AACjC;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,OAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,iBAAiB,kBAAkB,UAAU;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,eAAe;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;AACA,uBAAuB;AACvB,WAAW;AACX;AACA;AACA,2DAA2D,gBAAgB,IAAI;AAC/E;AACA;AACA;AACA;AACA;AACA,WAAW;AACX,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8DAA8D;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB;AACvB;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,WAAW;AACX,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,qBAAqB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA,yCAAyC,IAAI;AAC7C,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,aAAa,SAAS,YAAY;AAClD;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+EAA+E;AAC/E;;AAEA,+FAA+F,6BAA6B;AAC5H;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,iCAAiC;AACjC;AACA;AACA,+BAA+B;AAC/B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA,yDAAyD,WAAW;AACpE,6DAA6D,WAAW;AACxE;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,8CAA8C,gCAAgC;;AAE9E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,uEAAuE;AACvE;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA,CAAC;AACD;;;;;;;;;;;;;;;;;;ACrdgD;AACzC,iCAAiC,yDAAc;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,OAAO;AAChE;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qDAAqD,yBAAyB;AAC9E,qDAAqD,yBAAyB;AAC9E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;ACjOK;AACzC,kCAAkC,yDAAc;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;AC3BpC;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC3C8B;AACzE;AACA;AACA;AACA;AACA;AACA,QAAQ,uFAAM,+CAA+C,KAAK;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC;AACzC,YAAY,uFAAM,0CAA0C,YAAY,4BAA4B,IAAI;AACxG;AACA;AACA;AACA,6CAA6C;AAC7C,YAAY,uFAAM,gDAAgD,YAAY,6BAA6B,IAAI;AAC/G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB,kBAAkB,UAAU;AAC5B;AACA;AACA;AACA;AACA;AACA,wBAAwB,YAAY;AACpC;AACA,YAAY,uFAAM,uCAAuC,KAAK,uBAAuB,SAAS;AAC9F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,UAAU;AAC3B,kBAAkB,QAAQ;AAC1B;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA,YAAY,uFAAM,uCAAuC,KAAK,qBAAqB,MAAM;AACzF;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;AC1JpC;AACP;AACA;AACA;AACA;AACA,wBAAwB,cAAc;AACtC;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACgD;AACzC;AACP;AACO;AACP;AACA;AACA;AACA;AACO;AACP;AACA,+CAA+C,uEAA4B;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,iCAAiC,yDAAc;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC7C3C;AACA;AACO;AACP;AACA;AACA;AACA,CAAC,8BAA8B;AAC/B;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,0BAA0B;AACrC,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,SAAS;AACpB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,SAAS;AACpB;AACA;AACA;AACA;AACA,YAAY,QAAQ;AACpB;AACO,iMAAiM,+BAA+B;AACvO;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACzEL;AACS;AAC/C;AACO;AACP;AACA,sCAAsC;AACtC,mCAAmC;AACnC;AACA;AACA;AACA;AACA,8BAA8B;AAC9B,2BAA2B;AAC3B;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA,gBAAgB,QAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA,qCAAqC,+DAAqB;AAC1D,4BAA4B,+DAAqB;AACjD,8BAA8B,kEAAwB;AACtD;AACA,QAAQ,8EAAkC;AAC1C,gBAAgB,kEAAwB;AACxC;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;AC1DG;AACC;AAC/C;AACO,gCAAgC,wDAAY;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,2EAA4B;AACzD;AACA;AACA;AACA,8BAA8B;AAC9B,kCAAkC;AAClC,6BAA6B;AAC7B;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,gBAAgB,kBAAkB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,+BAA+B;AAC3D;AACA,4CAA4C,yFAA6C;AACzF;AACA;AACA,SAAS;AACT,yBAAyB,uEAA2B;AACpD;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,UAAU;AAC1B;AACA;AACA,8CAA8C;AAC9C,mCAAmC;AACnC,+CAA+C;AAC/C,+BAA+B;AAC/B;AACA;AACA,0BAA0B;AAC1B;AACA,wCAAwC;AACxC,yEAAyE;AACzE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,oFAAwC;AAC/D;AACA,0BAA0B,YAAY,EAAE,GAAG,EAAE,aAAa;AAC1D,SAAS;AACT,kBAAkB,OAAO,EAAE,2BAA2B,EAAE,QAAQ;AAChE;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,eAAe;AAC9B;AACA,gBAAgB,SAAS;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,eAAe;AAC9B,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,eAAe;AAC9B,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA,6BAA6B,qFAAyC;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,gCAAgC;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,eAAe;AAC9B;AACA,gBAAgB,WAAW;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,qFAAyC;AACzD,gBAAgB,yFAA6C;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,2BAA2B;AACvD;AACA;AACA;AACA,6BAA6B,oFAAwC;AACrE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,UAAU;AACzB,eAAe,eAAe;AAC9B,gBAAgB,0BAA0B;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACxNL;AACQ;AACvC;AACP,qBAAqB,2EAA4B;AACjD;AACA,qBAAqB,6DAAmB;AACxC;AACA;AACA;AACA,oBAAoB,wBAAwB;AAC5C;AACA;AACA;AACA;AACA,2BAA2B,mBAAmB;AAC9C;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACpBL;AACQ;AAC9C;AACO;AACP;AACA;AACA,wCAAwC,uDAAa;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,+BAA+B;AAC3D;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,+BAA+B;AAC3D;AACA;AACA;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,sCAAsC;AAClE;AACA,+CAA+C,iBAAiB,iBAAiB,cAAc;AAC/F,4BAA4B,sCAAsC;AAClE;AACA;AACA;AACA,4BAA4B,kEAAwB;AACpD;AACA,sBAAsB,+DAAgB,SAAS,sEAAuB;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oEAAoE,4EAA6B;AACjG;AACA;AACA,mBAAmB,uDAAa;AAChC;AACA;AACA;AACA,kBAAkB;AAClB,mBAAmB;AACnB,qBAAqB;AACrB,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,cAAc;AAC1E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,oBAAoB;AACpB,eAAe;AACf,cAAc;AACd,cAAc;AACd,kBAAkB;AAClB;AACA;AACA,gBAAgB,cAAc;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,iBAAiB,6BAA6B,WAAW;AAChG;AACA;AACA;AACA;AACA;AACA,eAAe,UAAU;AACzB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,4DAAkB;AAC5C,4BAA4B,kEAAwB;AACpD,yBAAyB,uDAAa;AACtC,yBAAyB,+DAAqB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,WAAW;AAC1B;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,WAAW;AAC1B,eAAe,WAAW;AAC1B,eAAe,WAAW;AAC1B,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,4DAAkB;AAC5C,4BAA4B,kEAAwB;AACpD,yBAAyB,uDAAa;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC3N3C;AACO;AACP;AACA;AACA;AACA;AACA,CAAC,oCAAoC;AACrC;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,cAAc;AAC7B,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AC7B3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AAClB;AACS;AAC4C;AACwB;AAClC;AACG;AAClB;AACM;AACtD;AACA;AACA;AACA;AACA,CAAC,wCAAwC;AACzC,WAAW,YAAY;AACvB,WAAW,QAAQ;AACnB,YAAY,UAAU;AACtB;AACA;AACA;AACA,eAAe,kDAAQ;AACvB;AACA,oEAA0B;AAC1B;AACA;AACO;AACP,wCAAwC,iDAAO;AAC/C;AACA;AACA;AACA;AACO;AACP,iBAAiB;AACjB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,UAAU;AACzB;AACA,+BAA+B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACO,sBAAsB,qDAAW;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,iDAAiD;AAC5D;AACA,WAAW,mDAAmD;AAC9D;AACA,WAAW,mDAAmD;AAC9D;AACA;AACA;AACA;AACA;AACA,8EAA8E,sCAAsC;AACpH,kFAAkF,sCAAsC;AACxH;AACA;AACA;AACA,gFAAgF,oCAAoC;AACpH;AACA;AACA;AACA;AACA,0FAA0F,iBAAiB;AAC3G,mFAAmF,wDAAwD;AAC3I;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,gDAAM;AACjC;AACA,0BAA0B,mDAAS;AACnC;AACA,wBAAwB,gDAAM;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,+BAA+B,2CAAc;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,+CAAQ;AAC7C;AACA;AACA;AACA,qBAAqB;AACrB,oBAAoB,yDAAe,CAAC,gDAAM,EAAE,gDAAM,IAAI,QAAQ,IAAI,gDAAM,IAAI,UAAU;AACtF;AACA;AACA,oBAAoB,yDAAe;AACnC;AACA,aAAa;AACb,+BAA+B,2CAAc;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA,aAAa;AACb,+BAA+B,0DAAgB;AAC/C;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,2BAA2B,sBAAsB,gBAAgB,uBAAuB;AACxF,6BAA6B,kBAAkB,kBAAkB,oBAAoB;AACrF;AACA,iBAAiB,cAAc,KAAK,UAAU,UAAU,0BAA0B,WAAW,aAAa;AAC1G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iDAAiD,uDAAa;AAC9D,qDAAqD,8EAA2B;AAChF;AACA,wCAAwC,+DAAY;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0DAA0D,SAAS;AACnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,eAAe,MAAM;AAC5E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B,SAAS;AACT;AACA;AACA,0CAA0C,cAAc,iCAAiC,cAAc;AACvG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA,0CAA0C,cAAc;AACxD,SAAS;AACT;AACA;AACA;AACA,sBAAsB;AACtB,SAAS;AACT;AACA,oCAAoC,iCAAiC;AACrE,0CAA0C,cAAc,iCAAiC,cAAc;AACvG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,cAAc;AACxD,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,yBAAyB;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,QAAQ,6IAA6I;AAC5M;AACA;AACA;AACA;AACA;AACA,gCAAgC,+DAAY;AAC5C;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,oBAAoB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,cAAc,2BAA2B,UAAU;AACvF;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA,gBAAgB,0DAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,oCAAoC,iDAAO;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8EAA8E,oBAAoB;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,4BAA4B;AAChD;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6C,EAAE,gCAAgC,2BAA2B;AAC1G;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,MAAM;AAC3C;AACA,iDAAiD,OAAO;AACxD;AACA;AACA,mDAAmD,aAAa;AAChE;AACA;AACA;AACA,+DAA+D,MAAM;AACrE,wCAAwC,OAAO;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD,cAAc;AACvE;AACA,yDAAyD,uBAAuB;AAChF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE,yCAAyC,uBAAuB;AAChE;AACA,2CAA2C,mBAAmB;AAC9D,2CAA2C,yBAAyB;AACpE;AACA;AACA,eAAe,WAAW;AAC1B,eAAe,SAAS;AACxB,gBAAgB,YAAY;AAC5B;AACA;AACA;AACA;AACA,iBAAiB,4EAAmC;AACpD,sBAAsB,sEAA6B;AACnD;AACA,iBAAiB,8EAAoC;AACrD,sBAAsB,0EAAgC;AACtD;AACA;AACA;AACA;AACA,iCAAiC,2DAAiB,IAAI,0EAAgC;AACtF,iCAAiC,yDAAgB,IAAI,sEAA6B;AAClF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,8DAAwB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA,mEAAmE,kBAAkB;AACrF,eAAe,WAAW;AAC1B,eAAe,QAAQ;AACvB,eAAe,cAAc;AAC7B,iBAAiB,aAAa,wBAAwB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,mFAAM;AAChC,8BAA8B,mFAAM;AACpC,eAAe,iGAAgB,sBAAsB,6FAAY,UAAU,6FAAY;AACvF;AACA;AACA;AACA,kBAAkB,cAAc;AAChC,mBAAmB,WAAW;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uDAAuD,kEAAwB;AAC/E;AACA,qCAAqC,uDAAa;AAClD;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,eAAe,KAAK;AACpB,gBAAgB,UAAU;AAC1B;AACA;AACA,oBAAoB,yCAAE;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mDAAmD,EAAE;AACrD;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB;AACA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD,OAAO;AACvD;AACA;AACA;AACA,4BAA4B,kEAAwB;AACpD,yDAAyD,kEAAwB,CAAC;AAClF,iCAAiC,uDAAa;AAC9C,qCAAqC,8EAA2B;AAChE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,yCAAE;AAC5B;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,qDAAqD,MAAM,QAAQ,sBAAsB;AACzF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC,IAAI,IAAI;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;AC7xB3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC4E;AACyB;AAC9F;AACP;AACA,8CAA8C,kHAAiC,GAAG,yCAAyC;AAC3H,wFAAwF,0FAAS;AACjG;AACA,KAAK;AACL;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;AClBD;AACY;AAC6B;AACU;AACtF;AACP;AACA,6BAA6B,uGAA0B;AACvD,KAAK;AACL;AACA,uBAAuB,yDAAW;AAClC,wBAAwB,gEAAW;AACnC,KAAK;AACL;AACA,oBAAoB,sGAA2B;AAC/C,gBAAgB,kGAAuB;AACvC,sBAAsB,wGAA6B;AACnD,0BAA0B,+GAAiC;AAC3D,kBAAkB,oGAAyB;AAC3C,sBAAsB,wGAA6B;AACnD,yBAAyB,8GAAgC;AACzD,0BAA0B,+GAAiC;AAC3D,kBAAkB,oGAAyB;AAC3C,iBAAiB,mGAAwB;AACzC,KAAK;AACL;AACO;AACP;AACA;AACA;AACA;AACA;AACA,CAAC,IAAI;AACE;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACO;AACP;AACA;AACA,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,gBAAgB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,oBAAoB;AACnC,gBAAgB,UAAU;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;AC7E3C,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AACuD;AACP;AACtF;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA,mCAAmC,kEAAwB;AAC3D;AACA,8BAA8B,oGAAqB;AACnD,gBAAgB,wBAAwB;AACxC;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,2BAA2B,4DAAkB;AAC7C;AACA,sBAAsB,kEAAwB,EAAE,4DAAkB;AAClE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,cAAc;AACtC,4BAA4B,kBAAkB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,4DAAkB,mBAAmB,sDAAsD;AAChH;AACA;AACA,qBAAqB,iEAAuB;AAC5C;AACA,kCAAkC,+DAAqB;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,oIAAoI;AACvK,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT,+BAA+B,mDAAS,IAAI,2BAA2B;AACvE,YAAY,qEAA2B;AACvC,SAAS;AACT;AACA,mCAAmC,wDAAc;AACjD;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,oBAAoB,qBAAM;AAC1B;AACA;AACA,oBAAoB,yDAAe,eAAe,0FAA0F;AAC5I;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,2CAA2C;AACvE;AACA,iBAAiB;AACjB;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,sBAAsB,UAAU;AAChC;AACA;AACA;AACA;AACA,gBAAgB,sDAAY;AAC5B,oBAAoB,oDAAU;AAC9B;AACA,iBAAiB,iDAAO,WAAW,kDAAQ;AAC3C,uCAAuC,oDAAU;AACjD,IAAI,sDAAY;AAChB;AACA;AACA;AACA;AACA,IAAI,kDAAQ;AACZ,yCAAyC,iGAAiG;AAC1I;AACA;AACA,QAAQ,sDAAY;AACpB,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kBAAkB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,MAAM,GAAG,QAAQ;AACnD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA,sBAAsB,0DAA0D;AAChF;AACA,oBAAoB,6DAAmB;AACvC;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,KAAK;AACL;AACO;AACP;AACA,wBAAwB,cAAc;AACtC;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACO;AACP,oBAAoB,aAAa;AACjC;AACA,4BAA4B,SAAS;AACrC,uCAAuC,2GAAyB;AAChE;AACA,oBAAoB,oEAA0B;AAC9C;AACA;AACA;AACO;AACP;AACA;AACA,0BAA0B,QAAQ,GAAG,WAAW;AAChD;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;ACzUpC;AACP;AACA,0CAA0C,4KAA2C;AACrF;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,8BAA8B,QAAQ,uBAAuB;AAC7D,sBAAsB,0CAA0C;AAChE;AACA,KAAK;AACL;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;ACdL;AACA;AACtC;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA,oBAAoB,oBAAoB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,uBAAuB,6CAAW;AAClC;AACA;AACA;AACA;AACA;AACA,wCAAwC,wDAAc;AACtD;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;AC5C3C;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACpBN;AAC9B;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACA;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kDAAS;AAC7B;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;;;;;;;;;;;;;;;;;;ACpL3C;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACO;AACP;AACA,2CAA2C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACpCE;AACK;AAClD;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,QAAQ;AACnB,YAAY;AACZ;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,aAAa;AACzB;AACA;AACA;AACA,2DAA2D,sDAAM;AACjE;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACO;AACP;AACA,4BAA4B,sDAAM;AAClC;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,sBAAsB,sDAAM;AAC5B,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACA;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACA;AACA;AACA,sBAAsB,sDAAM;AAC5B,oBAAoB,cAAc;AAClC;AACA;AACA;AACO;AACP;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;AACO;AACP;AACA;AACA;AACA,oBAAoB,eAAe;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA,oBAAoB,gBAAgB;AACpC,wBAAwB,gBAAgB;AACxC,2BAA2B,oDAAW;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,gBAAgB;AAC3B,YAAY,QAAQ;AACpB;AACO;AACP;AACA;AACA,oBAAoB,YAAY;AAChC,4BAA4B,YAAY;AACxC;AACA;AACA;AACA;AACA;AACA;AACA,yCAAyC,gBAAgB;AAClD;AACP;AACA;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB,WAAW,SAAS;AACpB,YAAY,UAAU;AACtB;AACO;AACP;AACA,oCAAoC;AACpC,iCAAiC;AACjC;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB,WAAW,QAAQ;AACnB,WAAW,oCAAoC;AAC/C;AACA,aAAa,UAAU;AACvB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,kDAAS;AAC7B;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,QAAQ;AACnB;AACO;AACP;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB,WAAW,OAAO;AAClB,YAAY;AACZ;AACO;AACP;AACA;AACA;AACA,2CAA2C;;;;;;;;;;ACxO3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ,KAA0B;AAClC;AACA,MAAM,SAAS,IAA0C;AACzD,QAAQ,mCAAO,aAAa,gBAAgB;AAAA,kGAAC;AAC7C,MAAM,KAAK,EAEN;AACL,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA,qBAAqB,QAAQ;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC;AACxC,wCAAwC;AACxC,2CAA2C;AAC3C,+BAA+B;AAC/B,8BAA8B;AAC9B,+BAA+B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,wBAAwB;AAChD;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,mBAAmB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,OAAO;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,uBAAuB;AAC/C;AACA;AACA;AACA;AACA;AACA,wBAAwB,uBAAuB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,wBAAwB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qDAAqD;AACrD;AACA,2BAA2B;AAC3B,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,YAAY;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA,wBAAwB,iBAAiB;AACzC;AACA;AACA,4BAA4B,SAAS;AACrC;AACA;AACA;AACA,wBAAwB,gCAAgC;AACxD;AACA;AACA;AACA,oBAAoB,SAAS;AAC7B;AACA;AACA,oBAAoB,OAAO;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;;;;;;;;;;;AC1aY;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc,OAAO;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;AACA,SAAS,eAAe;AACxB;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA;AACA,oBAAoB,OAAO;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,SAAS;AAC/B;AACA;AACA;AACA,kBAAkB,OAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,SAAS;AAC/B;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB,gBAAgB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;AClJa;AACb;AACA,8CAA6C,EAAE,aAAa,EAAC;AAC7D,mBAAmB,GAAG,YAAY;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B,8CAA8C,qCAAqC;AACnF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;;;;;;;;;;;;;;;;;;ACtFnB,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AAC1C;AACO;AACP;AACA,0BAA0B,6DAAmB;AAC7C;AACA;AACA,wBAAwB,qBAAqB;AAC7C,sCAAsC,sBAAsB;AAC5D;AACA,wBAAwB,iBAAiB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,oBAAoB,iBAAiB;AACrC;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC,mBAAmB,EAAE,mBAAmB;AAC7E;AACA,+BAA+B,8BAA8B;AAC7D;AACA;AACA;AACA,kCAAkC,cAAc,EAAE,uBAAuB;AACzE;AACA;AACA;AACA,+BAA+B,8BAA8B;AAC7D;AACA;AACA;AACA;AACA,kCAAkC,cAAc,EAAE,OAAO,EAAE,cAAc,EAAE,cAAc;AACzF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACpEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACA;AACA;AACA;AACA;AACA;AACA;AACP;AACO;AACA;;;;;;;;;;;;;;;;;ACtBP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB;AACxB,yBAAyB;AACzB,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,eAAe;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,eAAe;AACvC;AACA,4BAA4B,eAAe;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,4BAA4B,mBAAmB;AAC/C;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA,4BAA4B,mBAAmB;AAC/C;AACA;AACA;AACA;AACA;AACA,wBAAwB,gBAAgB;AACxC,4BAA4B,gBAAgB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;AChfA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACnC;AACP;AACA;AACA,kCAAkC,6DAAmB;AACrD,wBAAwB,qBAAqB;AAC7C,4BAA4B,wBAAwB;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA,2DAA2D;AAC3D;AACA;AACA,wBAAwB,WAAW;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,uDAAuD;AACvD;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,mBAAmB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,qBAAqB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D;AAC/D;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C,+FAA+F;AAC/F;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D;AAC/D;AACA;AACA;AACA;AACA,wBAAwB,mBAAmB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iFAAiF;AACjF;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,gBAAgB;AACvC;AACA;AACA,yBAAyB,oBAAoB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8CAA8C;AAC9C,gDAAgD;AAChD;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACrUiC;AACK;AACsC;AACE;AACL;AACF;AACjC;AAC2D;AAClB;AAC/E;AACA,+BAA+B,gGAAsB;AACrD;AACA;AACA;AACA;AACA,mBAAmB,iGAA6B;AAChD;AACA,mBAAmB,qGAAgC;AACnD;AACA,mBAAmB,qGAAgC;AACnD;AACA,mBAAmB,qGAAgC;AACnD;AACA;AACA,mBAAmB,yFAAwB;AAC3C;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,iBAAiB,EAAE,IAAI;AAC/C;AACA,KAAK;AACL;AACA;AACO,gDAAgD,6DAAmB;AAC1E,iBAAiB;AACjB,qBAAqB;AACrB,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA,wCAAwC,sGAAyB;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,6FAAmB;AAChD;AACA,mIAAmI,yDAAe,CAAC,gDAAM,wCAAwC,yDAAe;AAChN;AACA;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA,wDAAwD,qDAAe;AACvE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kDAAkD,uDAAa;AAC/D;AACA;AACA;AACA,6BAA6B,6FAAmB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,4BAA4B,qCAAqC;AACjE;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,qCAAqC;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,8FAAiB;AACzC;AACA,wBAAwB,0FAAa;AACrC;AACA;AACA;AACA;AACA;AACA,iBAAiB,oGAAmB;AACpC;AACA,SAAS;AACT;AACA;AACA;AACA;AACO,kCAAkC,6DAAmB;AAC5D,iBAAiB,OAAO,yDAAmB;AAC3C,qBAAqB,OAAO,yDAAmB;AAC/C,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA,qEAAqE,qDAAe;AACpF;AACA;AACA;AACA;AACA;AACA;AACO,kDAAkD,6DAAmB;AAC5E,iBAAiB;AACjB,qBAAqB,OAAO,0EAAoC;AAChE,0BAA0B;AAC1B,yBAAyB;AACzB;AACA;AACA;AACA,eAAe,0BAA0B;AACzC,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,QAAQ;AACvB,eAAe,aAAa;AAC5B,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oDAAoD,sDAAgB;AACpE,gDAAgD,uDAAa;AAC7D,yBAAyB,6FAAmB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB,yFAAwB;AAC9C;AACA;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA,8BAA8B,oGAAmB;AACjD,8BAA8B,oGAAmB;AACjD;AACA;AACA;AACA,2BAA2B,oGAAmB;AAC9C;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACtQO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,sCAAsC;AAChC;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,oBAAoB;AACd;AACP;AACA;AACA;AACA;AACA;AACA;AACA,CAAC,8BAA8B;AACxB;AACA;AACA;AACA;AACA;AACP;AACA;AACA,CAAC,sBAAsB;AAChB;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,YAAY,6GAA6G;AACzH,YAAY,wGAAwG;AACpH,WAAW,mGAAmG;AAC9G,YAAY,oEAAoE;AAChF;AACA;AACA;AACA,KAAK;AACL,WAAW,0DAA0D;AACrE;;;;;;;;;;;;;;;;;;;;;;;;ACtDA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACsC;AACI;AACjB;AACgE;AACzF;AACA;AACA;AACA;AACA;AACA,WAAW,WAAW;AACtB;AACO;AACP,0BAA0B,mGAAiB;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,wDAAc;AAC9C,2BAA2B,wDAAc;AACzC;AACA;AACA;AACA,YAAY,+CAAC;AACb;AACA,YAAY,+CAAC;AACb;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,wBAAwB,mDAAS;AACjC,iBAAiB,gDAAM;AACvB,YAAY,oDAAU;AACtB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,oBAAoB,gBAAgB;AACpC;AACA,+CAA+C,kBAAkB;AACjE;AACA;AACA,SAAS;AACT;AACA;AACO;AACP;AACA,8BAA8B,mGAAiB;AAC/C;AACA;AACA;AACA,cAAc,uEAA6B;AAC3C;AACA,KAAK;AACL;;;;;;;;;;;;;;;;;;;;;;;;ACjFA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAg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,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACsC;AACA;AACgC;AACtE,YAAY,gFAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,8BAA8B,qDAAW;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE,6DAA6D;AACpI,sEAAsE,6BAA6B;AACnG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,6FAA6F;AACjH,oBAAoB,6FAA6F;AACjH;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB;AACA;AACA,2BAA2B,0DAAgB;AAC3C;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,iCAAiC;AACjC,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD,4BAA4B;AAC5E,kDAAkD,6BAA6B;AAC/E;AACA,qEAAqE,4CAA4C;AACjH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,qEAAqE,0DAAgB;AACrF;AACA,SAAS;AACT;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uEAAuE,qCAAqC;AAC5G;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,6CAA6C,mCAAmC,YAAY,OAAO,EAAE,+BAA+B,GAAG;AACvI;AACA;AACA;AACA,iCAAiC,2BAA2B;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,kDAAQ;AACtC;AACA;AACA,gDAAgD,gDAAM;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,uCAAuC,gDAAM,gBAAgB,OAAO,EAAE,oBAAoB;AAC1F;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA,qBAAqB;AACrB;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA,kCAAkC,4CAA4C;AAC9E,wBAAwB,gDAAM,oBAAoB,UAAU,uBAAuB,GAAG;AACtF;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,6BAA6B,4BAA4B;AACzD,iCAAiC,6BAA6B;AAC9D;AACA,iEAAiE,UAAU;AAC3E;AACA;AACA;AACA;AACA;AACA;AACA,+CAA+C,IAAI,OAAO,KAAK;AAC/D;AACA;;;;;;;;;;;;;;;;;;;;;;;;ACjQA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AAC0C;AACJ;AACA;AACoB;AACN;AACpD;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,OAAO;AAClB,YAAY,oBAAoB;AAChC;AACO;AACP;AACA,mBAAmB,4EAAkC;AACrD;AACA;AACA;AACA;AACA,oCAAoC,sEAAwB;AAC5D,qCAAqC,gEAAW;AAChD;AACA,oCAAoC,6DAAmB,gBAAgB,YAAY;AACnF;AACA;AACA;AACA;AACA;AACA;AACA,4DAA4D,oBAAoB;AAChF,gCAAgC,gDAAM,OAAO,6CAA6C;AAC1F;AACA,uDAAuD,0BAA0B;AACjF;AACA,6CAA6C,YAAY;AACzD;AACA;AACA;AACA,iBAAiB;AACjB,+BAA+B,0DAAgB;AAC/C,8BAA8B,iDAAO;AACrC,6BAA6B,gDAAM;AACnC;AACA;AACA,6BAA6B,oDAAU;AACvC;AACA;AACA;AACA,yBAAyB,oDAAU;AACnC;AACA;AACA,mBAAmB,mDAAS;AAC5B,KAAK;AACL;;;;;;;;;;;;AC/DA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;UAEA;UACA;;;;;WCzBA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;WACA;WACA;WACA;WACA;;;;;WCJA;WACA;WACA;WACA;WACA,GAAG;WACH;WACA;WACA,CAAC;;;;;WCPD;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;WCNA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;WACA;;;;;WCfA;;WAEA;WACA;WACA;WACA;WACA;WACA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;WAEA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACrBA,iBAAiB,SAAI,IAAI,SAAI;AAC7B,4BAA4B,+DAA+D,iBAAiB;AAC5G;AACA,oCAAoC,MAAM,+BAA+B,YAAY;AACrF,mCAAmC,MAAM,mCAAmC,YAAY;AACxF,gCAAgC;AAChC;AACA,KAAK;AACL;AACA;AAC0C;AACJ;AACA;AAC/B,qBAAqB,oDAAU;AAC2D;AAC1B;AACT;AACE;AAChB;AAC8B;AACS;AAC+B;AACE;AACrE;AACuB;AAChC;AACwB;AACa;AACI;AACG;AACH;AAC5C;AACuB;AAC9D;AACO;AACP;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,mFAAiC;AAChD;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,mFAAiC;AAChD;AACA;AACA;AACA;AACA,QAAQ,4DAAkB;AAC1B;AACA;AACO;AACP;AACA;AACA,mBAAmB,0FAAY;AAC/B,wBAAwB,kEAAwB;AAChD,QAAQ,4DAAkB;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,wCAAwC,EAAE,gCAAgC,EAAE,gBAAgB;AACxH,qBAAqB,MAAM,2DAA2D,YAAY;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,wCAAwC,EAAE,gCAAgC,EAAE,eAAe;AAC3H,yBAAyB,MAAM,2DAA2D,YAAY;AACtG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACO;AACP,wBAAwB,yDAAiB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,iFAAO;AACtB;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,uEAAe;AAC9B;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA;AACA,6BAA6B;AACtB;AACP;AACA;AACA;AACA,2BAA2B,6DAAc;AACzC;AACA;AACA,0BAA0B,4EAAqB;AAC/C;AACA,qBAAqB,4CAA4C;AACjE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,qGAAiB,iHAAiH,kEAAwB,QAAQ,gEAAa,EAAE,kFAAoB,EAAE,kFAAoB,EAAE,oFAAsB;AAC5Q;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,gCAAgC;AAChC,6BAA6B;AAC7B,mCAAmC;AACnC;AACO;AACP;AACA;AACA;AACA,2BAA2B,6DAAc;AACzC;AACA;AACA,+BAA+B,4EAAqB;AACpD,mCAAmC,kEAAwB;AAC3D,8BAA8B,qGAAqB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,oEAAa;AACpD;AACA;AACA;AACA;AACA,8BAA8B,4DAAkB;AAChD;AACA;AACA;AACA,4BAA4B,0DAAgB;AAC5C;AACA,yCAAyC,qEAAqE;AAC9G;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AACxB;AACP;AACA,YAAY,sDAAY,GAAG,yCAAyC;AACpE,YAAY,4DAAkB;AAC9B;AACA;AACA;AACA;AACA,gEAAgE,iEAAsB;AACtF;AACA,4BAA4B,iEAAkB;AAC9C,6BAA6B,iEAAW;AACxC,oBAAoB,+DAAqB;AACzC,sBAAsB,6DAAmB;AACzC,iBAAiB,uDAAa;AAC9B;AACA,cAAc,uEAA6B;AAC3C,KAAK;AACL;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACO;AACP;AACA,qBAAqB,sDAAY,GAAG,sDAAsD;AAC1F;AACA;AACA,8DAA8D,SAAS;AACvE,6BAA6B,6EAAS;AACtC;AACA;AACA,cAAc,uEAA6B;AAC3C;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,2BAA2B;AAC3B;AACO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACO;AACP;AACA;AACA,mBAAmB,uDAAa;AAChC;AACA;AACA;AACA,+BAA+B,kEAAwB;AACvD;AACA,8BAA8B,0FAAY;AAC1C;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,uDAAuD,8BAA8B;AACrF,YAAY,wEAA8B,WAAW,2DAAiB;AACtE,SAAS;AACT;AACA;AACA,YAAY,0DAAgB;AAC5B;AACA;AACA;AACA;AACA,6BAA6B,wDAAc;AAC3C,YAAY,mDAAS;AACrB;AACA;AACA,aAAa;AACb,qBAAqB,gDAAM;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACO;AACP;AACA,mCAAmC,0EAA2B;AAC9D,KAAK;AACL;AACA;AACA;AACA,6BAA6B;AAC7B;AACO;AACP;AACA,gEAAgE,iEAAsB;AACtF;AACA,eAAe,gFAAqB;AACpC,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,oBAAoB,8FAAgB;AACpC;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACd;AACP,IAAI,wDAAO;AACX;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,qEAAmB;AAClC;AACA;AACA;AACA;AACA;AACO;AACP,eAAe,qFAAmC;AAClD;AACA;AACA,sBAAsB;AACtB;AACO;AACP;AACA,mBAAmB,4EAAkC;AACrD,+BAA+B,8DAAoB;AACnD;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,oEAA0B;AAC5D,2BAA2B,8DAAoB;AAC/C;AACA,0CAA0C,6DAAmB,8BAA8B,UAAU;AACrG,oCAAoC,kEAAwB;AAC5D,gDAAgD,cAAc,YAAY,SAAS;AACnF,wCAAwC,QAAQ,WAAW,0BAA0B;AACrF;AACA;AACA;AACA;AACA,+CAA+C,WAAW,uDAAa,EAAE;AACzE,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,4EAA4E,kBAAkB;AAC9F;AACA;AACA,QAAQ,yDAAe;AACvB;AACA,sBAAsB,kEAAwB;AAC9C,qDAAqD,KAAK;AAC1D;AACA,KAAK;AACL;AACA;AACA;AACA,qBAAqB;AACd;AACP,oBAAoB,uGAAyB,WAAW,kDAAK;AAC7D,eAAe,0DAAgB;AAC/B,mBAAmB,kGAAqB;AACxC;AACA;AACA;AACA,yBAAyB,gEAAmB;AAC5C,sBAAsB,+DAAqB,EAAE,gEAAmB;AAChE,sBAAsB,4DAAe,aAAa,4DAAe;AACjE;AACA,IAAI,uEAA6B;AACjC;AACA;AACA,qBAAqB;AACd;AACP,kBAAkB,0FAAgB,YAAY,gGAAsB;AACpE;AACA;AACO;AACP;AACA,QAAQ,6DAAmB;AAC3B;AACA,2BAA2B,6DAAmB,uBAAuB,YAAY;AACjF,eAAe,4EAAe;AAC9B,KAAK;AACL","sources":["webpack://bio/./node_modules/@biowasm/aioli/dist/aioli.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/aminoacids.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/nucleotides.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/seq-palettes.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/sequence-encoder.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/unknown.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/cell-renderer.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/fasta-handler.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/notation-converter.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/splitter.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/utils/units-handler.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/vd-regions.js","webpack://bio/./node_modules/@datagrok-libraries/bio/src/viewers/web-logo.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/sequence-space.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/typed-metrics.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/viewers/activity-cliffs.js","webpack://bio/./node_modules/@datagrok-libraries/ml/src/workers/dimensionality-reducing-worker-creator.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/dataframe-utils.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/random.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/similarity-metrics.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/type-declarations.js","webpack://bio/./node_modules/@datagrok-libraries/utils/src/vector-operations.js","webpack://bio/./node_modules/fast-sha256/sha256.js","webpack://bio/./node_modules/fastest-levenshtein/index.js","webpack://bio/./node_modules/jaro-winkler-typescript/lib/index.js","webpack://bio/./src/calculations/fingerprints.ts","webpack://bio/./src/const.ts","webpack://bio/./src/seq_align.ts","webpack://bio/./src/utils/atomic-works.ts","webpack://bio/./src/utils/cell-renderer.ts","webpack://bio/./src/utils/constants.ts","webpack://bio/./src/utils/convert.ts","webpack://bio/./src/utils/multiple-sequence-alignment.ts","webpack://bio/./src/utils/sequence-activity-cliffs.ts","webpack://bio/./src/utils/sequence-space.ts","webpack://bio/./src/utils/utils.ts","webpack://bio/./src/viewers/vd-regions-viewer.ts","webpack://bio/./src/widgets/representations.ts","webpack://bio/external var \"$\"","webpack://bio/external var \"DG\"","webpack://bio/external var \"grok\"","webpack://bio/external var \"rxjs\"","webpack://bio/external var \"ui\"","webpack://bio/external var \"wu\"","webpack://bio/webpack/bootstrap","webpack://bio/webpack/runtime/compat get default export","webpack://bio/webpack/runtime/define property getters","webpack://bio/webpack/runtime/get javascript chunk filename","webpack://bio/webpack/runtime/global","webpack://bio/webpack/runtime/hasOwnProperty shorthand","webpack://bio/webpack/runtime/make namespace object","webpack://bio/webpack/runtime/publicPath","webpack://bio/webpack/runtime/jsonp chunk loading","webpack://bio/./src/package.ts"],"sourcesContent":["(function (global, factory) {\n  typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n  typeof define === 'function' && define.amd ? define(factory) :\n  (global = global || self, global.Aioli = factory());\n}(this, (function () { 'use strict';\n\n  var name = \"@biowasm/aioli\";\n  var version = \"2.4.0-rc2\";\n  var description = \"A framework for building WebAssembly-based genomics tools\";\n  var browser = \"dist/aioli.js\";\n  var worker = \"dist/aioli.worker.js\";\n  var main = \"aioli.js\";\n  var repository = {\n  \ttype: \"git\",\n  \turl: \"git+https://github.com/biowasm/aioli.git\"\n  };\n  var author = \"Robert Aboukhalil\";\n  var license = \"MIT\";\n  var dependencies = {\n  \tcomlink: \"^4.3.1\",\n  \t\"wasm-feature-detect\": \"^1.2.11\"\n  };\n  var devDependencies = {\n  \t\"@rollup/plugin-commonjs\": \"^11.0.1\",\n  \t\"@rollup/plugin-json\": \"^4.1.0\",\n  \t\"@rollup/plugin-node-resolve\": \"^7.0.0\",\n  \tcypress: \"^8.0.0\",\n  \trollup: \"^1.29.0\",\n  \t\"rollup-plugin-terser\": \"^7.0.2\"\n  };\n  var scripts = {\n  \tbuild: \"rollup -c\",\n  \tdev: \"rollup -c -w\",\n  \ttest: \"cypress run\",\n  \t\"test-open\": \"cypress open\"\n  };\n  var files = [\n  \t\"dist\"\n  ];\n  var pkg = {\n  \tname: name,\n  \tversion: version,\n  \tdescription: description,\n  \tbrowser: browser,\n  \tworker: worker,\n  \tmain: main,\n  \trepository: repository,\n  \tauthor: author,\n  \tlicense: license,\n  \tdependencies: dependencies,\n  \tdevDependencies: devDependencies,\n  \tscripts: scripts,\n  \tfiles: files\n  };\n\n  /**\r\n   * Copyright 2019 Google Inc. All Rights Reserved.\r\n   * Licensed under the Apache License, Version 2.0 (the \"License\");\r\n   * you may not use this file except in compliance with the License.\r\n   * You may obtain a copy of the License at\r\n   *     http://www.apache.org/licenses/LICENSE-2.0\r\n   * Unless required by applicable law or agreed to in writing, software\r\n   * distributed under the License is distributed on an \"AS IS\" BASIS,\r\n   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r\n   * See the License for the specific language governing permissions and\r\n   * limitations under the License.\r\n   */\r\n  const proxyMarker = Symbol(\"Comlink.proxy\");\r\n  const createEndpoint = Symbol(\"Comlink.endpoint\");\r\n  const releaseProxy = Symbol(\"Comlink.releaseProxy\");\r\n  const throwMarker = Symbol(\"Comlink.thrown\");\r\n  const isObject = (val) => (typeof val === \"object\" && val !== null) || typeof val === \"function\";\r\n  /**\r\n   * Internal transfer handle to handle objects marked to proxy.\r\n   */\r\n  const proxyTransferHandler = {\r\n      canHandle: (val) => isObject(val) && val[proxyMarker],\r\n      serialize(obj) {\r\n          const { port1, port2 } = new MessageChannel();\r\n          expose(obj, port1);\r\n          return [port2, [port2]];\r\n      },\r\n      deserialize(port) {\r\n          port.start();\r\n          return wrap(port);\r\n      },\r\n  };\r\n  /**\r\n   * Internal transfer handler to handle thrown exceptions.\r\n   */\r\n  const throwTransferHandler = {\r\n      canHandle: (value) => isObject(value) && throwMarker in value,\r\n      serialize({ value }) {\r\n          let serialized;\r\n          if (value instanceof Error) {\r\n              serialized = {\r\n                  isError: true,\r\n                  value: {\r\n                      message: value.message,\r\n                      name: value.name,\r\n                      stack: value.stack,\r\n                  },\r\n              };\r\n          }\r\n          else {\r\n              serialized = { isError: false, value };\r\n          }\r\n          return [serialized, []];\r\n      },\r\n      deserialize(serialized) {\r\n          if (serialized.isError) {\r\n              throw Object.assign(new Error(serialized.value.message), serialized.value);\r\n          }\r\n          throw serialized.value;\r\n      },\r\n  };\r\n  /**\r\n   * Allows customizing the serialization of certain values.\r\n   */\r\n  const transferHandlers = new Map([\r\n      [\"proxy\", proxyTransferHandler],\r\n      [\"throw\", throwTransferHandler],\r\n  ]);\r\n  function expose(obj, ep = self) {\r\n      ep.addEventListener(\"message\", function callback(ev) {\r\n          if (!ev || !ev.data) {\r\n              return;\r\n          }\r\n          const { id, type, path } = Object.assign({ path: [] }, ev.data);\r\n          const argumentList = (ev.data.argumentList || []).map(fromWireValue);\r\n          let returnValue;\r\n          try {\r\n              const parent = path.slice(0, -1).reduce((obj, prop) => obj[prop], obj);\r\n              const rawValue = path.reduce((obj, prop) => obj[prop], obj);\r\n              switch (type) {\r\n                  case \"GET\" /* GET */:\r\n                      {\r\n                          returnValue = rawValue;\r\n                      }\r\n                      break;\r\n                  case \"SET\" /* SET */:\r\n                      {\r\n                          parent[path.slice(-1)[0]] = fromWireValue(ev.data.value);\r\n                          returnValue = true;\r\n                      }\r\n                      break;\r\n                  case \"APPLY\" /* APPLY */:\r\n                      {\r\n                          returnValue = rawValue.apply(parent, argumentList);\r\n                      }\r\n                      break;\r\n                  case \"CONSTRUCT\" /* CONSTRUCT */:\r\n                      {\r\n                          const value = new rawValue(...argumentList);\r\n                          returnValue = proxy(value);\r\n                      }\r\n                      break;\r\n                  case \"ENDPOINT\" /* ENDPOINT */:\r\n                      {\r\n                          const { port1, port2 } = new MessageChannel();\r\n                          expose(obj, port2);\r\n                          returnValue = transfer(port1, [port1]);\r\n                      }\r\n                      break;\r\n                  case \"RELEASE\" /* RELEASE */:\r\n                      {\r\n                          returnValue = undefined;\r\n                      }\r\n                      break;\r\n                  default:\r\n                      return;\r\n              }\r\n          }\r\n          catch (value) {\r\n              returnValue = { value, [throwMarker]: 0 };\r\n          }\r\n          Promise.resolve(returnValue)\r\n              .catch((value) => {\r\n              return { value, [throwMarker]: 0 };\r\n          })\r\n              .then((returnValue) => {\r\n              const [wireValue, transferables] = toWireValue(returnValue);\r\n              ep.postMessage(Object.assign(Object.assign({}, wireValue), { id }), transferables);\r\n              if (type === \"RELEASE\" /* RELEASE */) {\r\n                  // detach and deactive after sending release response above.\r\n                  ep.removeEventListener(\"message\", callback);\r\n                  closeEndPoint(ep);\r\n              }\r\n          });\r\n      });\r\n      if (ep.start) {\r\n          ep.start();\r\n      }\r\n  }\r\n  function isMessagePort(endpoint) {\r\n      return endpoint.constructor.name === \"MessagePort\";\r\n  }\r\n  function closeEndPoint(endpoint) {\r\n      if (isMessagePort(endpoint))\r\n          endpoint.close();\r\n  }\r\n  function wrap(ep, target) {\r\n      return createProxy(ep, [], target);\r\n  }\r\n  function throwIfProxyReleased(isReleased) {\r\n      if (isReleased) {\r\n          throw new Error(\"Proxy has been released and is not useable\");\r\n      }\r\n  }\r\n  function createProxy(ep, path = [], target = function () { }) {\r\n      let isProxyReleased = false;\r\n      const proxy = new Proxy(target, {\r\n          get(_target, prop) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              if (prop === releaseProxy) {\r\n                  return () => {\r\n                      return requestResponseMessage(ep, {\r\n                          type: \"RELEASE\" /* RELEASE */,\r\n                          path: path.map((p) => p.toString()),\r\n                      }).then(() => {\r\n                          closeEndPoint(ep);\r\n                          isProxyReleased = true;\r\n                      });\r\n                  };\r\n              }\r\n              if (prop === \"then\") {\r\n                  if (path.length === 0) {\r\n                      return { then: () => proxy };\r\n                  }\r\n                  const r = requestResponseMessage(ep, {\r\n                      type: \"GET\" /* GET */,\r\n                      path: path.map((p) => p.toString()),\r\n                  }).then(fromWireValue);\r\n                  return r.then.bind(r);\r\n              }\r\n              return createProxy(ep, [...path, prop]);\r\n          },\r\n          set(_target, prop, rawValue) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              // FIXME: ES6 Proxy Handler `set` methods are supposed to return a\r\n              // boolean. To show good will, we return true asynchronously ¯\\_(ツ)_/¯\r\n              const [value, transferables] = toWireValue(rawValue);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"SET\" /* SET */,\r\n                  path: [...path, prop].map((p) => p.toString()),\r\n                  value,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n          apply(_target, _thisArg, rawArgumentList) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              const last = path[path.length - 1];\r\n              if (last === createEndpoint) {\r\n                  return requestResponseMessage(ep, {\r\n                      type: \"ENDPOINT\" /* ENDPOINT */,\r\n                  }).then(fromWireValue);\r\n              }\r\n              // We just pretend that `bind()` didn’t happen.\r\n              if (last === \"bind\") {\r\n                  return createProxy(ep, path.slice(0, -1));\r\n              }\r\n              const [argumentList, transferables] = processArguments(rawArgumentList);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"APPLY\" /* APPLY */,\r\n                  path: path.map((p) => p.toString()),\r\n                  argumentList,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n          construct(_target, rawArgumentList) {\r\n              throwIfProxyReleased(isProxyReleased);\r\n              const [argumentList, transferables] = processArguments(rawArgumentList);\r\n              return requestResponseMessage(ep, {\r\n                  type: \"CONSTRUCT\" /* CONSTRUCT */,\r\n                  path: path.map((p) => p.toString()),\r\n                  argumentList,\r\n              }, transferables).then(fromWireValue);\r\n          },\r\n      });\r\n      return proxy;\r\n  }\r\n  function myFlat(arr) {\r\n      return Array.prototype.concat.apply([], arr);\r\n  }\r\n  function processArguments(argumentList) {\r\n      const processed = argumentList.map(toWireValue);\r\n      return [processed.map((v) => v[0]), myFlat(processed.map((v) => v[1]))];\r\n  }\r\n  const transferCache = new WeakMap();\r\n  function transfer(obj, transfers) {\r\n      transferCache.set(obj, transfers);\r\n      return obj;\r\n  }\r\n  function proxy(obj) {\r\n      return Object.assign(obj, { [proxyMarker]: true });\r\n  }\r\n  function toWireValue(value) {\r\n      for (const [name, handler] of transferHandlers) {\r\n          if (handler.canHandle(value)) {\r\n              const [serializedValue, transferables] = handler.serialize(value);\r\n              return [\r\n                  {\r\n                      type: \"HANDLER\" /* HANDLER */,\r\n                      name,\r\n                      value: serializedValue,\r\n                  },\r\n                  transferables,\r\n              ];\r\n          }\r\n      }\r\n      return [\r\n          {\r\n              type: \"RAW\" /* RAW */,\r\n              value,\r\n          },\r\n          transferCache.get(value) || [],\r\n      ];\r\n  }\r\n  function fromWireValue(value) {\r\n      switch (value.type) {\r\n          case \"HANDLER\" /* HANDLER */:\r\n              return transferHandlers.get(value.name).deserialize(value.value);\r\n          case \"RAW\" /* RAW */:\r\n              return value.value;\r\n      }\r\n  }\r\n  function requestResponseMessage(ep, msg, transfers) {\r\n      return new Promise((resolve) => {\r\n          const id = generateUUID();\r\n          ep.addEventListener(\"message\", function l(ev) {\r\n              if (!ev.data || !ev.data.id || ev.data.id !== id) {\r\n                  return;\r\n              }\r\n              ep.removeEventListener(\"message\", l);\r\n              resolve(ev.data);\r\n          });\r\n          if (ep.start) {\r\n              ep.start();\r\n          }\r\n          ep.postMessage(Object.assign({ id }, msg), transfers);\r\n      });\r\n  }\r\n  function generateUUID() {\r\n      return new Array(4)\r\n          .fill(0)\r\n          .map(() => Math.floor(Math.random() * Number.MAX_SAFE_INTEGER).toString(16))\r\n          .join(\"-\");\r\n  }\n  //# sourceMappingURL=comlink.mjs.map\n\n  // Constants\n  const URL_CDN_ROOT = \"https://cdn.biowasm.com/v2\";\n  const CONFIG_DEFAULTS = {\n  \t// Biowasm CDN URLs\n  \turlCDN: URL_CDN_ROOT,\n  \t// Get the Worker code corresponding to the current Aioli version\n  \turlAioli: `${URL_CDN_ROOT}/aioli/${pkg.version}/aioli.worker.js`,\n  \t// Where we can find the base biowasm module (only modify this for local development)\n  \turlBaseModule: null,\n\n  \t// Folder to use for mounting the shared filesystem\n  \tdirShared: \"/shared\",\n  \t// Folder to use for mounting File/Blob objects to the virtual file system\n  \tdirMounted: \"/mnt\",\n  \t// Folder to use for symlinks (basically, we make a symlink to each file mounted on WORKERFS\n  \t// so that operations like \"samtools index\" don't crash due to the read-only nature of WORKERS).\n  \t// Also mount URLs lazily in that folder.\n  \tdirData: \"/data\",\n  \t// Interleave stdout/stderr. If set to false, `.exec()` returns an object { \"stdout\": <text>, \"stderr\": <text> }\n  \tprintInterleaved: true,\n\n  \t// Callback function to run whenever we receive a message from the WebWorker with payload { type: \"biowasm\", value: ... }.\n  \t// See <https://github.com/biowasm/biowasm/tree/main/tools/bhtsne> for an example of how this can be used to send regular updates\n  \t// back to the main thread before callMain() is done running.\n  \tcallback: null,\n\n  \t// Debugging\n  \tdebug: false,\n  \tenv: \"prd\"\n  };\n\n  // Class: 1 object = 1 worker; user can decide if they want tools running in separate threads or all of them in one\n  class Aioli\n  {\n  \tconstructor(tools, config={})\n  \t{\n  \t\tif(tools == null)\n  \t\t\tthrow \"Expecting array of tools as input to Aioli constructor.\";\n\n  \t\t// Parse user input\n  \t\tif(!Array.isArray(tools))\n  \t\t\ttools = [ tools ];\n  \t\t// Overwrite default config if specified\n  \t\tconfig = Object.assign({}, CONFIG_DEFAULTS, config);\n  \t\t// For convenience, support \"<tool>/<version>\" or \"<tool>/<program>/<version>\" instead of object config\n  \t\ttools = tools.map(this._parseTool);\n  \t\t// If testing with different environment e.g. cdn-stg.biowasm.com\n  \t\tif(config.env != \"prd\") {\n  \t\t\tconfig.urlCDN = config.urlCDN.replace(\"cdn\", `cdn-${config.env}`);\n  \t\t\tconfig.urlAioli = config.urlAioli.replace(\"cdn\", `cdn-${config.env}`);\n  \t\t}\n\n  \t\t// Add biowasm base module to list of tools to initialize (need this for the shared virtual filesystem)\n  \t\ttools = [{\n  \t\t\ttool: \"base\",\n  \t\t\tversion: pkg.version,\n  \t\t\turlPrefix: config.urlBaseModule\n  \t\t}, ...tools];\n\n  \t\t// Set state\n  \t\tthis.tools = tools;\n  \t\tthis.config = config;\n\n  \t\t// Handle callback (delete it because we can't send a function to the WebWorker)\n  \t\tif(this.config.callback != null)\n  \t\t\tthis.callback = this.config.callback;\n  \t\tdelete this.config.callback;\n\n  \t\treturn this.init();\n  \t}\n\n  \t// Initialize the WebWorker and the WebAssembly modules within it\n  \tasync init() {\n  \t\t// Note: We can only create a WebWorker using a URL that has the same origin as the app running it, i.e. we\n  \t\t// can't do `new Worker(\"https://cdn.biowasm.com/v2/...\")`, so instead we need to fetch the contents of that\n  \t\t// URL first, and then create the WebWorker. See <https://stackoverflow.com/a/60015898>.\n  \t\tconst moduleJS = await (await fetch(this.config.urlAioli)).text();\n  \t\tconst moduleBlob = new Blob([moduleJS], { type: \"application/javascript\" });\n\n  \t\t// Create the WebWorker\n  \t\tconst worker = new Worker(URL.createObjectURL(moduleBlob));\n  \t\t// Listen for \"biowasm\" messages from the WebWorker\n  \t\tif(this.callback)\n  \t\t\tworker.onmessage = e => {\n  \t\t\t\tif(e.data.type == \"biowasm\")\n  \t\t\t\t\tthis.callback(e.data.value);\n  \t\t\t};\n\n  \t\tconst aioli = wrap(worker);\n  \t\taioli.tools = this.tools;\n  \t\taioli.config = this.config;\n\n  \t\t// Initialize the tools inside the WebWorker\n  \t\tawait aioli.init();\n\n  \t\treturn aioli;\n  \t}\n\n  \t// Parse \"<tool>/<version>\" and \"<tool>/<program>/<version>\" into { \"tool\": <tool>, \"program\": <program>, \"version\": <version> }\n  \t_parseTool(tool)\n  \t{\n  \t\t// If not a string, leave it as is\n  \t\tif(typeof tool !== \"string\")\n  \t\t\treturn tool;\n\n  \t\t// Support \"<tool>/<version>\" and \"<tool>/<program>/<version>\"\n  \t\tconst toolSplit = tool.split(\"/\");\n  \t\tif(toolSplit.length != 2 && toolSplit.length != 3)\n  \t\t\tthrow \"Expecting '<tool>/<version>' or '<tool>/<program>/<version>'\";\n\n  \t\treturn {\n  \t\t\ttool: toolSplit[0],\n  \t\t\tprogram: toolSplit.length == 3 ? toolSplit[1] : toolSplit[0],\n  \t\t\tversion: toolSplit[toolSplit.length - 1]\n  \t\t};\n  \t}\n  }\n\n  return Aioli;\n\n})));\n//# sourceMappingURL=aioli.js.map\n","import { SeqPaletteBase } from './seq-palettes';\nexport class AminoacidsPalettes extends SeqPaletteBase {\n    static get Lesk() {\n        if (this.lesk === void 0) {\n            this.lesk = this.makePalette([\n                [['G', 'A', 'S', 'T'], 'orange'],\n                [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],\n                [['N', 'Q', 'H'], 'magenta'],\n                [['D', 'E'], 'red'],\n                [['K', 'R'], 'all_blue'],\n            ], false, AminoacidsPalettes);\n        }\n        return this.lesk;\n    }\n    static get GrokGroups() {\n        if (this.grokGroups === void 0) {\n            this.grokGroups = this.makePalette([\n                [['C', 'U'], 'yellow'],\n                [['G', 'P'], 'red'],\n                [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],\n                [['R', 'H', 'K'], 'light_blue'],\n                [['D', 'E'], 'dark_blue'],\n                [['S', 'T', 'N', 'Q'], 'orange'],\n            ], false, AminoacidsPalettes);\n        }\n        return this.grokGroups;\n    }\n    static get RasMol() {\n        if (this.rasMol === void 0) {\n            this.rasMol = new AminoacidsPalettes({\n                // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm\n                'D': '#E60A0A',\n                'E': '#E60A0A',\n                'C': '#E6E600',\n                'M': '#E6E600',\n                'K': '#145AFF',\n                'R': '#145AFF',\n                'S': '#FA9600',\n                'T': '#FA9600',\n                'F': '#3232AA',\n                'Y': '#3232AA',\n                'N': '#00DCDC',\n                'Q': '#00DCDC',\n                'G': '#EBEBEB',\n                'L': '#0F820F',\n                'V': '#0F820F',\n                'I': '#0F820F',\n                'A': '#C8C8C8',\n                'W': '#B45AB4',\n                'H': '#8282D2',\n                'P': '#DC9682',\n                'others': '#BEA06E',\n            });\n        }\n        return this.rasMol;\n    }\n}\nexport class Aminoacids {\n    static getPalette(scheme = 'grok') {\n        switch (scheme) {\n            case 'grok':\n                return AminoacidsPalettes.GrokGroups;\n            case 'lesk':\n                return AminoacidsPalettes.Lesk;\n            default:\n                throw new Error(`ChemPalette: scheme \\`${scheme}\\` does not exist`);\n        }\n    }\n    /**\n     * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted\n     *\n     * @param {string} c raw amino\n     * @return {[string, string]} outer and inner content\n     */\n    static getInnerOuter(c) {\n        let isInner = 0;\n        let inner = '';\n        let outer = '';\n        for (const char of c) {\n            if (char == '(')\n                isInner++;\n            else if (char == ')')\n                isInner--;\n            else if (isInner)\n                inner += char;\n            else\n                outer += char;\n        }\n        return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];\n    }\n    static getColorAAPivot(monomer = '', scheme = 'grok') {\n        var _a, _b, _c;\n        //const chemPaletteInstance = AAPalettes.GrokGroups();\n        const chemPaletteInstance = this.getPalette(scheme);\n        let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);\n        outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);\n        innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);\n        if (monomer.length == 1 || monomer[1] == '(') {\n            const amino = (_a = monomer[0]) === null || _a === void 0 ? void 0 : _a.toUpperCase();\n            return amino in chemPaletteInstance ?\n                [chemPaletteInstance.get(amino), amino, innerMonomer, 1] :\n                [this.undefinedColor, outerMonomer, innerMonomer, 1];\n        }\n        if (monomer[0] == 'd' && monomer[1] in chemPaletteInstance) {\n            if (monomer.length == 2 || monomer[2] == '(') {\n                const amino = (_b = monomer[1]) === null || _b === void 0 ? void 0 : _b.toUpperCase();\n                return amino in chemPaletteInstance ?\n                    [chemPaletteInstance.get(amino), amino, innerMonomer, 2] :\n                    [this.undefinedColor, outerMonomer, innerMonomer, 2];\n            }\n        }\n        if (monomer.substring(0, 3) in this.AAFullNames) {\n            if (monomer.length == 3 || monomer[3] == '(') {\n                const amino = this.AAFullNames[monomer.substring(0, 3)];\n                return amino in chemPaletteInstance ?\n                    [chemPaletteInstance.get(amino), amino, innerMonomer, 3] :\n                    [this.undefinedColor, outerMonomer, innerMonomer, 3];\n            }\n        }\n        if (((_c = monomer[0]) === null || _c === void 0 ? void 0 : _c.toLowerCase()) == monomer[0]) {\n            if (monomer.substring(1, 3) in this.AAFullNames) {\n                if (monomer.length == 4 || monomer[4] == '(') {\n                    const amino = this.AAFullNames[monomer.substring(1, 3)];\n                    return amino in chemPaletteInstance ?\n                        [chemPaletteInstance.get(amino), amino, innerMonomer, 4] :\n                        [this.undefinedColor, outerMonomer, innerMonomer, 4];\n                }\n            }\n        }\n        return [this.undefinedColor, outerMonomer, innerMonomer, 0];\n    }\n}\nAminoacids.SemType = 'Aminoacids';\nAminoacids.SemTypeMultipleAlignment = 'AminoacidsMultipleAlignment';\nAminoacids.undefinedColor = 'rgb(100,100,100)';\nAminoacids.Names = {\n    'G': 'Glycine',\n    'L': 'Leucine',\n    'Y': 'Tyrosine',\n    'S': 'Serine',\n    'E': 'Glutamic acid',\n    'Q': 'Glutamine',\n    'D': 'Aspartic acid',\n    'N': 'Asparagine',\n    'F': 'Phenylalanine',\n    'A': 'Alanine',\n    'K': 'Lysine',\n    'R': 'Arginine',\n    'H': 'Histidine',\n    'C': 'Cysteine',\n    'V': 'Valine',\n    'P': 'Proline',\n    'W': 'Tryptophan',\n    'I': 'Isoleucine',\n    'M': 'Methionine',\n    'T': 'Threonine',\n};\nAminoacids.AASmiles = {\n    'G': 'NCC(=O)O',\n    'L': 'N[C@H](CC(C)C)C(=O)O',\n    'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',\n    'S': 'NC(CO)C(=O)O',\n    'E': 'N[C@@H](CCC(O)=O)C(=O)O',\n    'Q': 'N[C@@H](CCC(N)=O)C(=O)O',\n    'D': 'N[C@@H](CC(O)=O)C(=O)O',\n    'N': 'N[C@@H](CC(N)=O)C(=O)O',\n    'F': 'NC(CC1=CC=CC=C1)C(=O)O',\n    'A': 'N[C@H](C)C(=O)O',\n    'K': 'NC(CCCCN)C(=O)O',\n    'R': 'N[C@H](CCCNC(=N)C)C(=O)O',\n    'H': 'NC(CC1=CN=C[N]1)C(=O)O',\n    'C': 'N[C@@H](CS)C(=O)O',\n    'V': 'NC(C(C)C)C(=O)O',\n    'P': 'N(CCC1)C1C(=O)O',\n    'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',\n    'I': 'N[C@H]([C@H](C)CC)C(=O)O',\n    'M': 'NC(CCSC)C(=O)O',\n    'T': 'NC(C(O)C)C(=O)O',\n};\nAminoacids.AASmilesTruncated = {\n    'G': '*C*',\n    'L': 'CC(C)C[C@H](*)*',\n    'Y': 'C1=CC(=CC=C1CC(*)*)O',\n    'S': 'OCC(*)C*',\n    'E': '*[C@@H](CCC(O)=O)*',\n    'Q': '*N[C@@H](CCC(N)=O)*',\n    'D': '*[C@@H](CC(O)=O)*',\n    'N': '*[C@@H](CC(N)=O)*',\n    'F': 'C1=CC=C(C=C1)CC(*)*',\n    'A': 'C[C@H](*)*',\n    'K': 'C(CCN)CC(*)*',\n    'R': '*[C@H](CCCNC(=N)C)*',\n    'H': 'C1=C(NC=N1)CC(*)*',\n    'C': 'C([C@@H](*)*)S',\n    'V': 'CC(C)C(*)*',\n    'P': 'C1CCN(*)C1*',\n    'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',\n    'I': 'CC[C@H](C)[C@H](*)*',\n    'M': 'CSCCC(*)*',\n    'T': 'CC(O)C(*)*',\n};\n/** TODO: Full?\n */\nAminoacids.AAFullNames = {\n    'Ala': 'A',\n    'Arg': 'R',\n    'Asn': 'N',\n    'Asp': 'D',\n    'Cys': 'C',\n    'Gln': 'Q',\n    'Glu': 'E',\n    'Gly': 'G',\n    'His': 'H',\n    'Ile': 'I',\n    'Leu': 'L',\n    'Lys': 'K',\n    'Met': 'M',\n    'Phe': 'F',\n    'Pro': 'P',\n    'Ser': 'S',\n    'Thr': 'T',\n    'Trp': 'W',\n    'Tyr': 'Y',\n    'Val': 'V',\n};\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aminoacids.js","sourceRoot":"","sources":["aminoacids.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 {StringDictionary} from '@datagrok-libraries/utils/src/type-declarations';\nimport {SeqPalette, SeqPaletteBase} from './seq-palettes';\n\nexport class AminoacidsPalettes extends SeqPaletteBase {\n  private static lesk: SeqPalette;\n\n  public static get Lesk(): SeqPalette {\n    if (this.lesk === void 0) {\n      this.lesk = this.makePalette([\n        [['G', 'A', 'S', 'T'], 'orange'],\n        [['C', 'V', 'I', 'L', 'P', 'F', 'Y', 'M', 'W'], 'all_green'],\n        [['N', 'Q', 'H'], 'magenta'],\n        [['D', 'E'], 'red'],\n        [['K', 'R'], 'all_blue'],\n      ], false, AminoacidsPalettes);\n    }\n    return this.lesk;\n  }\n\n  private static grokGroups: SeqPalette;\n\n  public static get GrokGroups(): SeqPalette {\n    if (this.grokGroups === void 0) {\n      this.grokGroups = this.makePalette([\n        [['C', 'U'], 'yellow'],\n        [['G', 'P'], 'red'],\n        [['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'], 'all_green'],\n        [['R', 'H', 'K'], 'light_blue'],\n        [['D', 'E'], 'dark_blue'],\n        [['S', 'T', 'N', 'Q'], 'orange'],\n      ], false, AminoacidsPalettes);\n    }\n    return this.grokGroups;\n  }\n\n  private static rasMol: SeqPalette;\n\n  public static get RasMol(): SeqPalette {\n    if (this.rasMol === void 0) {\n      this.rasMol = new AminoacidsPalettes({\n        // http://acces.ens-lyon.fr/biotic/rastop/help/colour.htm\n        'D': '#E60A0A', // asp, aspartic acid, asp\n        'E': '#E60A0A', // glu, glutamic acid\n        'C': '#E6E600', // cys, cysteine\n        'M': '#E6E600', // met, methionine\n        'K': '#145AFF', // lys, lysine\n        'R': '#145AFF', // arg, arginine\n        'S': '#FA9600', // ser, serine\n        'T': '#FA9600', // thr, threonine\n        'F': '#3232AA', // phe, phenylalanine\n        'Y': '#3232AA', // tyr, tyrosine\n        'N': '#00DCDC', // asn, asparagine\n        'Q': '#00DCDC', // gln, glutamine\n        'G': '#EBEBEB', // gly, glycine\n        'L': '#0F820F', // leu, leucine\n        'V': '#0F820F', // val, valine\n        'I': '#0F820F', // ile, isoleucine\n        'A': '#C8C8C8', // ala, alanine\n        'W': '#B45AB4', // trp, tryptophan\n        'H': '#8282D2', // his, histidine\n        'P': '#DC9682', // pro, proline\n        'others': '#BEA06E',\n      });\n    }\n    return this.rasMol;\n  }\n}\n\nexport class Aminoacids {\n  static readonly SemType: string = 'Aminoacids';\n\n  static readonly SemTypeMultipleAlignment: string = 'AminoacidsMultipleAlignment';\n\n  static undefinedColor = 'rgb(100,100,100)';\n\n  public static Names: StringDictionary = {\n    'G': 'Glycine',\n    'L': 'Leucine',\n    'Y': 'Tyrosine',\n    'S': 'Serine',\n    'E': 'Glutamic acid',\n    'Q': 'Glutamine',\n    'D': 'Aspartic acid',\n    'N': 'Asparagine',\n    'F': 'Phenylalanine',\n    'A': 'Alanine',\n    'K': 'Lysine',\n    'R': 'Arginine',\n    'H': 'Histidine',\n    'C': 'Cysteine',\n    'V': 'Valine',\n    'P': 'Proline',\n    'W': 'Tryptophan',\n    'I': 'Isoleucine',\n    'M': 'Methionine',\n    'T': 'Threonine',\n  };\n\n  public static AASmiles: StringDictionary = {\n    'G': 'NCC(=O)O',\n    'L': 'N[C@H](CC(C)C)C(=O)O',\n    'Y': 'NC(CC1=CC=C(O)C=C1)C(=O)O',\n    'S': 'NC(CO)C(=O)O',\n    'E': 'N[C@@H](CCC(O)=O)C(=O)O',\n    'Q': 'N[C@@H](CCC(N)=O)C(=O)O',\n    'D': 'N[C@@H](CC(O)=O)C(=O)O',\n    'N': 'N[C@@H](CC(N)=O)C(=O)O',\n    'F': 'NC(CC1=CC=CC=C1)C(=O)O',\n    'A': 'N[C@H](C)C(=O)O',\n    'K': 'NC(CCCCN)C(=O)O',\n    'R': 'N[C@H](CCCNC(=N)C)C(=O)O',\n    'H': 'NC(CC1=CN=C[N]1)C(=O)O',\n    'C': 'N[C@@H](CS)C(=O)O',\n    'V': 'NC(C(C)C)C(=O)O',\n    'P': 'N(CCC1)C1C(=O)O',\n    'W': 'N[C@@H](Cc1c2ccccc2n([H])c1)C(=O)O',\n    'I': 'N[C@H]([C@H](C)CC)C(=O)O',\n    'M': 'NC(CCSC)C(=O)O',\n    'T': 'NC(C(O)C)C(=O)O',\n  };\n\n  public static AASmilesTruncated: StringDictionary = {\n    'G': '*C*',\n    'L': 'CC(C)C[C@H](*)*',\n    'Y': 'C1=CC(=CC=C1CC(*)*)O',\n    'S': 'OCC(*)C*',\n    'E': '*[C@@H](CCC(O)=O)*',\n    'Q': '*N[C@@H](CCC(N)=O)*',\n    'D': '*[C@@H](CC(O)=O)*',\n    'N': '*[C@@H](CC(N)=O)*',\n    'F': 'C1=CC=C(C=C1)CC(*)*',\n    'A': 'C[C@H](*)*',\n    'K': 'C(CCN)CC(*)*',\n    'R': '*[C@H](CCCNC(=N)C)*',\n    'H': 'C1=C(NC=N1)CC(*)*',\n    'C': 'C([C@@H](*)*)S',\n    'V': 'CC(C)C(*)*',\n    'P': 'C1CCN(*)C1*',\n    'W': '*[C@@H](Cc1c2ccccc2n([H])c1)*',\n    'I': 'CC[C@H](C)[C@H](*)*',\n    'M': 'CSCCC(*)*',\n    'T': 'CC(O)C(*)*',\n  };\n\n  /** TODO: Full?\n   */\n  public static AAFullNames: StringDictionary = {\n    'Ala': 'A',\n    'Arg': 'R',\n    'Asn': 'N',\n    'Asp': 'D',\n    'Cys': 'C',\n    'Gln': 'Q',\n    'Glu': 'E',\n    'Gly': 'G',\n    'His': 'H',\n    'Ile': 'I',\n    'Leu': 'L',\n    'Lys': 'K',\n    'Met': 'M',\n    'Phe': 'F',\n    'Pro': 'P',\n    'Ser': 'S',\n    'Thr': 'T',\n    'Trp': 'W',\n    'Tyr': 'Y',\n    'Val': 'V',\n  };\n\n  public static getPalette(scheme: string = 'grok'): SeqPalette {\n    switch (scheme) {\n    case 'grok':\n      return AminoacidsPalettes.GrokGroups;\n    case 'lesk':\n      return AminoacidsPalettes.Lesk;\n    default:\n      throw new Error(`ChemPalette: scheme \\`${scheme}\\` does not exist`);\n    }\n  }\n\n  /**\n   * Returns divided amino acid with its content in the bracket, if the content is number, then its omitted\n   *\n   * @param {string} c raw amino\n   * @return {[string, string]} outer and inner content\n   */\n  public static getInnerOuter(c: string): [string, string] {\n    let isInner = 0;\n    let inner = '';\n    let outer = '';\n\n    for (const char of c) {\n      if (char == '(')\n        isInner++;\n      else if (char == ')')\n        isInner--;\n      else if (isInner)\n        inner += char;\n      else\n        outer += char;\n    }\n\n    return !isNaN(parseInt(inner)) ? [outer, ''] : [outer, inner];\n  }\n\n  public static getColorAAPivot(monomer: string = '', scheme: 'grok' = 'grok'): [string, string, string, number] {\n    //const chemPaletteInstance = AAPalettes.GrokGroups();\n    const chemPaletteInstance = this.getPalette(scheme);\n    let [outerMonomer, innerMonomer] = this.getInnerOuter(monomer);\n    outerMonomer = (outerMonomer.length > 6 ? `${outerMonomer.slice(0, 3)}...` : outerMonomer);\n    innerMonomer = (innerMonomer.length > 6 ? `${innerMonomer.slice(0, 3)}...` : innerMonomer);\n\n    if (monomer.length == 1 || monomer[1] == '(') {\n      const amino = monomer[0]?.toUpperCase()!;\n      return amino in chemPaletteInstance ?\n        [chemPaletteInstance.get(amino), amino, innerMonomer, 1] :\n        [this.undefinedColor, outerMonomer, innerMonomer, 1];\n    }\n\n    if (monomer[0] == 'd' && monomer[1]! in chemPaletteInstance) {\n      if (monomer.length == 2 || monomer[2] == '(') {\n        const amino = monomer[1]?.toUpperCase()!;\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance.get(amino), amino, innerMonomer, 2] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 2];\n      }\n    }\n\n    if (monomer.substring(0, 3) in this.AAFullNames) {\n      if (monomer.length == 3 || monomer[3] == '(') {\n        const amino = this.AAFullNames[monomer.substring(0, 3)];\n        return amino in chemPaletteInstance ?\n          [chemPaletteInstance.get(amino), amino, innerMonomer, 3] :\n          [this.undefinedColor, outerMonomer, innerMonomer, 3];\n      }\n    }\n\n    if (monomer[0]?.toLowerCase() == monomer[0]) {\n      if (monomer.substring(1, 3) in this.AAFullNames) {\n        if (monomer.length == 4 || monomer[4] == '(') {\n          const amino = this.AAFullNames[monomer.substring(1, 3)];\n          return amino in chemPaletteInstance ?\n            [chemPaletteInstance.get(amino), amino, innerMonomer, 4] :\n            [this.undefinedColor, outerMonomer, innerMonomer, 4];\n        }\n      }\n    }\n\n    return [this.undefinedColor, outerMonomer, innerMonomer, 0];\n  }\n}\n"]}","import { SeqPaletteBase } from './seq-palettes';\nexport class NucleotidesPalettes extends SeqPaletteBase {\n    static get Chromatogram() {\n        if (this.chromatogram === void 0) {\n            this.chromatogram = new NucleotidesPalettes({\n                'A': 'green',\n                'C': 'blue',\n                'G': 'black',\n                'T': 'red',\n                'U': 'red',\n                'others': 'gray',\n            });\n        }\n        return this.chromatogram;\n    }\n}\nexport class Nucleotides {\n}\nNucleotides.SemType = 'Nucleotides';\nNucleotides.SemTypeMultipleAlignment = 'NucleotidesMultipleAlignment';\nNucleotides.Names = {\n    'A': 'Adenine',\n    'C': 'Cytosine',\n    'G': 'Guanine',\n    'T': 'Thymine',\n    'U': 'Uracil',\n};\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVjbGVvdGlkZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJudWNsZW90aWRlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLQSxPQUFPLEVBQWEsY0FBYyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFMUQsTUFBTSxPQUFPLG1CQUFvQixTQUFRLGNBQWM7SUFHOUMsTUFBTSxLQUFLLFlBQVk7UUFDNUIsSUFBSSxJQUFJLENBQUMsWUFBWSxLQUFLLEtBQUssQ0FBQyxFQUFFO1lBQ2hDLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxtQkFBbUIsQ0FBQztnQkFDMUMsR0FBRyxFQUFFLE9BQU87Z0JBQ1osR0FBRyxFQUFFLE1BQU07Z0JBQ1gsR0FBRyxFQUFFLE9BQU87Z0JBQ1osR0FBRyxFQUFFLEtBQUs7Z0JBQ1YsR0FBRyxFQUFFLEtBQUs7Z0JBQ1YsUUFBUSxFQUFFLE1BQU07YUFDakIsQ0FBQyxDQUFDO1NBQ0o7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUM7SUFDM0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxPQUFPLFdBQVc7O0FBQ04sbUJBQU8sR0FBVyxhQUFhLENBQUM7QUFFaEMsb0NBQXdCLEdBQVcsOEJBQThCLENBQUM7QUFFcEUsaUJBQUssR0FBcUI7SUFDdEMsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsVUFBVTtJQUNmLEdBQUcsRUFBRSxTQUFTO0lBQ2QsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsUUFBUTtDQUNkLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBncm9rIGZyb20gJ2RhdGFncm9rLWFwaS9ncm9rJztcbmltcG9ydCAqIGFzIHVpIGZyb20gJ2RhdGFncm9rLWFwaS91aSc7XG5pbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG5pbXBvcnQge1N0cmluZ0RpY3Rpb25hcnl9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3R5cGUtZGVjbGFyYXRpb25zJztcbmltcG9ydCB7U2VxUGFsZXR0ZSwgU2VxUGFsZXR0ZUJhc2V9IGZyb20gJy4vc2VxLXBhbGV0dGVzJztcblxuZXhwb3J0IGNsYXNzIE51Y2xlb3RpZGVzUGFsZXR0ZXMgZXh0ZW5kcyBTZXFQYWxldHRlQmFzZSB7XG4gIHByaXZhdGUgc3RhdGljIGNocm9tYXRvZ3JhbTogU2VxUGFsZXR0ZTtcblxuICBwdWJsaWMgc3RhdGljIGdldCBDaHJvbWF0b2dyYW0oKTogU2VxUGFsZXR0ZSB7XG4gICAgaWYgKHRoaXMuY2hyb21hdG9ncmFtID09PSB2b2lkIDApIHtcbiAgICAgIHRoaXMuY2hyb21hdG9ncmFtID0gbmV3IE51Y2xlb3RpZGVzUGFsZXR0ZXMoe1xuICAgICAgICAnQSc6ICdncmVlbicsXG4gICAgICAgICdDJzogJ2JsdWUnLFxuICAgICAgICAnRyc6ICdibGFjaycsIC8vIG9yYW5nZSA/XG4gICAgICAgICdUJzogJ3JlZCcsXG4gICAgICAgICdVJzogJ3JlZCcsXG4gICAgICAgICdvdGhlcnMnOiAnZ3JheScsXG4gICAgICB9KTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuY2hyb21hdG9ncmFtO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBOdWNsZW90aWRlcyB7XG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlOiBzdHJpbmcgPSAnTnVjbGVvdGlkZXMnO1xuXG4gIHN0YXRpYyByZWFkb25seSBTZW1UeXBlTXVsdGlwbGVBbGlnbm1lbnQ6IHN0cmluZyA9ICdOdWNsZW90aWRlc011bHRpcGxlQWxpZ25tZW50JztcblxuICBwdWJsaWMgc3RhdGljIE5hbWVzOiBTdHJpbmdEaWN0aW9uYXJ5ID0ge1xuICAgICdBJzogJ0FkZW5pbmUnLFxuICAgICdDJzogJ0N5dG9zaW5lJyxcbiAgICAnRyc6ICdHdWFuaW5lJyxcbiAgICAnVCc6ICdUaHltaW5lJyxcbiAgICAnVSc6ICdVcmFjaWwnLFxuICB9O1xufVxuIl19","export class SeqPaletteBase {\n    constructor(palette) {\n        this._palette = palette;\n    }\n    static makePalette(dt, simplified = false, PaletteType = SeqPaletteBase) {\n        const palette = {};\n        dt.forEach((cp) => {\n            const objList = cp[0];\n            const colour = cp[1];\n            objList.forEach((obj, ind) => {\n                palette[obj] = this.colourPalette[colour][simplified ? 0 : ind];\n            });\n        });\n        return new PaletteType(palette);\n    }\n    get(m) {\n        return this._palette[m];\n    }\n}\nSeqPaletteBase.undefinedColor = 'rgb(100,100,100)';\n/** Palette with shades of primary colors */\nSeqPaletteBase.colourPalette = {\n    'orange': ['rgb(255,187,120)', 'rgb(245,167,100)', 'rgb(235,137,70)', 'rgb(205, 111, 71)'],\n    'all_green': ['rgb(44,160,44)', 'rgb(74,160,74)', 'rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)',\n        'rgb(24,110,79)', 'rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],\n    'all_blue': ['rgb(31,119,180)', 'rgb(23,190,207)', 'rgb(122, 102, 189)', 'rgb(158,218,229)', 'rgb(141, 124, 217)',\n        'rgb(31, 120, 150)'],\n    'magenta': ['rgb(162,106,192)', 'rgb(197,165,224)', 'rgb(208,113,218)'],\n    'red': ['rgb(214,39,40)', 'rgb(255,152,150)'],\n    'st_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(31,119,180)'],\n    'dark_blue': ['rgb(31,119,180)', 'rgb(31, 120, 150)'],\n    'light_blue': ['rgb(23,190,207)', 'rgb(158,218,229)', 'rgb(108, 218, 229)', 'rgb(23,190,227)'],\n    'lilac_blue': ['rgb(124,102,211)', 'rgb(149,134,217)', 'rgb(97, 81, 150)'],\n    'dark_green': ['rgb(23,103,57)', 'rgb(30,110,96)', 'rgb(60,131,95)', 'rgb(24,110,79)'],\n    'green': ['rgb(44,160,44)', 'rgb(74,160,74)'],\n    'light_green': ['rgb(152,223,138)', 'rgb(182, 223, 138)', 'rgb(152, 193, 138)'],\n    'st_green': ['rgb(44,160,44)', 'rgb(152,223,138)', 'rgb(39, 174, 96)', 'rgb(74,160,74)'],\n    'pink': ['rgb(247,182,210)'],\n    'brown': ['rgb(140,86,75)', 'rgb(102, 62, 54)'],\n    'gray': ['rgb(127,127,127)', 'rgb(199,199,199)', 'rgb(196,156,148)', 'rgb(222, 222, 180)'],\n    'yellow': ['rgb(188,189,34)'],\n    'white': ['rgb(230,230,230)'],\n};\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxLXBhbGV0dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic2VxLXBhbGV0dGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWlCQSxNQUFNLE9BQU8sY0FBYztJQTJDekIsWUFBWSxPQUFnQztRQUMxQyxJQUFJLENBQUMsUUFBUSxHQUFHLE9BQU8sQ0FBQztJQUMxQixDQUFDO0lBbEJTLE1BQU0sQ0FBQyxXQUFXLENBQUMsRUFBd0IsRUFDbkQsVUFBVSxHQUFHLEtBQUssRUFBRSxjQUFxQyxjQUFjO1FBRXZFLE1BQU0sT0FBTyxHQUE4QixFQUFFLENBQUM7UUFDOUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxDQUFDLEVBQUUsRUFBRSxFQUFFO1lBQ2hCLE1BQU0sT0FBTyxHQUFHLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN0QixNQUFNLE1BQU0sR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDckIsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsRUFBRTtnQkFDM0IsT0FBTyxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ2xFLENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7UUFDSCxPQUFPLElBQUksV0FBVyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFRTSxHQUFHLENBQUMsQ0FBUztRQUNsQixPQUFPLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDMUIsQ0FBQzs7QUFoRGEsNkJBQWMsR0FBRyxrQkFBa0IsQ0FBQztBQUVsRCw0Q0FBNEM7QUFDOUIsNEJBQWEsR0FBZ0M7SUFDekQsUUFBUSxFQUFFLENBQUMsa0JBQWtCLEVBQUUsa0JBQWtCLEVBQUUsaUJBQWlCLEVBQUUsbUJBQW1CLENBQUM7SUFDMUYsV0FBVyxFQUFFLENBQUMsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCO1FBQ3BHLGdCQUFnQixFQUFFLGtCQUFrQixFQUFFLG9CQUFvQixFQUFFLG9CQUFvQixDQUFDO0lBQ25GLFVBQVUsRUFBRSxDQUFDLGlCQUFpQixFQUFFLGlCQUFpQixFQUFFLG9CQUFvQixFQUFFLGtCQUFrQixFQUFFLG9CQUFvQjtRQUMvRyxtQkFBbUIsQ0FBQztJQUN0QixTQUFTLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsQ0FBQztJQUN2RSxLQUFLLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsQ0FBQztJQUM3QyxTQUFTLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxrQkFBa0IsRUFBRSxpQkFBaUIsQ0FBQztJQUNyRSxXQUFXLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxtQkFBbUIsQ0FBQztJQUNyRCxZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxpQkFBaUIsQ0FBQztJQUM5RixZQUFZLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsQ0FBQztJQUMxRSxZQUFZLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxnQkFBZ0IsRUFBRSxnQkFBZ0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUN0RixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUM3QyxhQUFhLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxvQkFBb0IsQ0FBQztJQUMvRSxVQUFVLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxnQkFBZ0IsQ0FBQztJQUN4RixNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztJQUM1QixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxrQkFBa0IsQ0FBQztJQUMvQyxNQUFNLEVBQUUsQ0FBQyxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsQ0FBQztJQUMxRixRQUFRLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztJQUM3QixPQUFPLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztDQUM5QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgZ3JvayBmcm9tICdkYXRhZ3Jvay1hcGkvZ3Jvayc7XG5pbXBvcnQgKiBhcyB1aSBmcm9tICdkYXRhZ3Jvay1hcGkvdWknO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuZXhwb3J0IGludGVyZmFjZSBTZXFQYWxldHRlIHtcbiAgLy8gVGhlcmUgYXJlIHRvbyBtdWNoIHByb2JsZW0gd2l0aCBpbmRleGVyIHByb3BlcnR5IGluIHR5cGVzY3JpcHQuXG4gIC8vIC8qKlxuICAvLyAgKiBAcGFyYW0ge3N0cmluZ30gbSBNb25vbWVyIGNoYXJhY3RlclxuICAvLyAgKiBAcmV0dXJuIHtzdHJpbmd9IENvbG9yXG4gIC8vICAqL1xuICAvLyBbbTogc3RyaW5nXTogc3RyaW5nO1xuICAvKiogTW9ub21lciBjb2xvclxuICAgKiBAcGFyYW0ge3N0cmluZ30gbSBNb25vbWVyXG4gICAqL1xuICBnZXQobTogc3RyaW5nKTogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgU2VxUGFsZXR0ZUJhc2UgaW1wbGVtZW50cyBTZXFQYWxldHRlIHtcbiAgcHVibGljIHN0YXRpYyB1bmRlZmluZWRDb2xvciA9ICdyZ2IoMTAwLDEwMCwxMDApJztcblxuICAvKiogUGFsZXR0ZSB3aXRoIHNoYWRlcyBvZiBwcmltYXJ5IGNvbG9ycyAqL1xuICBwdWJsaWMgc3RhdGljIGNvbG91clBhbGV0dGU6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nW10gfSA9IHtcbiAgICAnb3JhbmdlJzogWydyZ2IoMjU1LDE4NywxMjApJywgJ3JnYigyNDUsMTY3LDEwMCknLCAncmdiKDIzNSwxMzcsNzApJywgJ3JnYigyMDUsIDExMSwgNzEpJ10sXG4gICAgJ2FsbF9ncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDc0LDE2MCw3NCknLCAncmdiKDIzLDEwMyw1NyknLCAncmdiKDMwLDExMCw5NiknLCAncmdiKDYwLDEzMSw5NSknLFxuICAgICAgJ3JnYigyNCwxMTAsNzkpJywgJ3JnYigxNTIsMjIzLDEzOCknLCAncmdiKDE4MiwgMjIzLCAxMzgpJywgJ3JnYigxNTIsIDE5MywgMTM4KSddLFxuICAgICdhbGxfYmx1ZSc6IFsncmdiKDMxLDExOSwxODApJywgJ3JnYigyMywxOTAsMjA3KScsICdyZ2IoMTIyLCAxMDIsIDE4OSknLCAncmdiKDE1OCwyMTgsMjI5KScsICdyZ2IoMTQxLCAxMjQsIDIxNyknLFxuICAgICAgJ3JnYigzMSwgMTIwLCAxNTApJ10sXG4gICAgJ21hZ2VudGEnOiBbJ3JnYigxNjIsMTA2LDE5MiknLCAncmdiKDE5NywxNjUsMjI0KScsICdyZ2IoMjA4LDExMywyMTgpJ10sXG4gICAgJ3JlZCc6IFsncmdiKDIxNCwzOSw0MCknLCAncmdiKDI1NSwxNTIsMTUwKSddLFxuICAgICdzdF9ibHVlJzogWydyZ2IoMjMsMTkwLDIwNyknLCAncmdiKDE1OCwyMTgsMjI5KScsICdyZ2IoMzEsMTE5LDE4MCknXSxcbiAgICAnZGFya19ibHVlJzogWydyZ2IoMzEsMTE5LDE4MCknLCAncmdiKDMxLCAxMjAsIDE1MCknXSxcbiAgICAnbGlnaHRfYmx1ZSc6IFsncmdiKDIzLDE5MCwyMDcpJywgJ3JnYigxNTgsMjE4LDIyOSknLCAncmdiKDEwOCwgMjE4LCAyMjkpJywgJ3JnYigyMywxOTAsMjI3KSddLFxuICAgICdsaWxhY19ibHVlJzogWydyZ2IoMTI0LDEwMiwyMTEpJywgJ3JnYigxNDksMTM0LDIxNyknLCAncmdiKDk3LCA4MSwgMTUwKSddLFxuICAgICdkYXJrX2dyZWVuJzogWydyZ2IoMjMsMTAzLDU3KScsICdyZ2IoMzAsMTEwLDk2KScsICdyZ2IoNjAsMTMxLDk1KScsICdyZ2IoMjQsMTEwLDc5KSddLFxuICAgICdncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDc0LDE2MCw3NCknXSxcbiAgICAnbGlnaHRfZ3JlZW4nOiBbJ3JnYigxNTIsMjIzLDEzOCknLCAncmdiKDE4MiwgMjIzLCAxMzgpJywgJ3JnYigxNTIsIDE5MywgMTM4KSddLFxuICAgICdzdF9ncmVlbic6IFsncmdiKDQ0LDE2MCw0NCknLCAncmdiKDE1MiwyMjMsMTM4KScsICdyZ2IoMzksIDE3NCwgOTYpJywgJ3JnYig3NCwxNjAsNzQpJ10sXG4gICAgJ3BpbmsnOiBbJ3JnYigyNDcsMTgyLDIxMCknXSxcbiAgICAnYnJvd24nOiBbJ3JnYigxNDAsODYsNzUpJywgJ3JnYigxMDIsIDYyLCA1NCknXSxcbiAgICAnZ3JheSc6IFsncmdiKDEyNywxMjcsMTI3KScsICdyZ2IoMTk5LDE5OSwxOTkpJywgJ3JnYigxOTYsMTU2LDE0OCknLCAncmdiKDIyMiwgMjIyLCAxODApJ10sXG4gICAgJ3llbGxvdyc6IFsncmdiKDE4OCwxODksMzQpJ10sXG4gICAgJ3doaXRlJzogWydyZ2IoMjMwLDIzMCwyMzApJ10sXG4gIH07XG5cbiAgcHJvdGVjdGVkIHN0YXRpYyBtYWtlUGFsZXR0ZShkdDogW3N0cmluZ1tdLCBzdHJpbmddW10sXG4gICAgc2ltcGxpZmllZCA9IGZhbHNlLCBQYWxldHRlVHlwZTogdHlwZW9mIFNlcVBhbGV0dGVCYXNlID0gU2VxUGFsZXR0ZUJhc2VcbiAgKSB7XG4gICAgY29uc3QgcGFsZXR0ZTogeyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfSA9IHt9O1xuICAgIGR0LmZvckVhY2goKGNwKSA9PiB7XG4gICAgICBjb25zdCBvYmpMaXN0ID0gY3BbMF07XG4gICAgICBjb25zdCBjb2xvdXIgPSBjcFsxXTtcbiAgICAgIG9iakxpc3QuZm9yRWFjaCgob2JqLCBpbmQpID0+IHtcbiAgICAgICAgcGFsZXR0ZVtvYmpdID0gdGhpcy5jb2xvdXJQYWxldHRlW2NvbG91cl1bc2ltcGxpZmllZCA/IDAgOiBpbmRdO1xuICAgICAgfSk7XG4gICAgfSk7XG4gICAgcmV0dXJuIG5ldyBQYWxldHRlVHlwZShwYWxldHRlKTtcbiAgfVxuXG4gIHByaXZhdGUgcmVhZG9ubHkgX3BhbGV0dGU6IHsgW206IHN0cmluZ106IHN0cmluZyB9O1xuXG4gIGNvbnN0cnVjdG9yKHBhbGV0dGU6IHsgW206IHN0cmluZ106IHN0cmluZyB9KSB7XG4gICAgdGhpcy5fcGFsZXR0ZSA9IHBhbGV0dGU7XG4gIH1cblxuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuX3BhbGV0dGVbbV07XG4gIH1cbn1cbiJdfQ==","import { assert } from '@datagrok-libraries/utils/src/vector-operations';\nclass SideChainScales {\n    static getAvailableScales() {\n        return Object.entries(this.scales).map(([k, _]) => k);\n    }\n    static getScale(name) {\n        assert(!(this.scales[name] === undefined), `Scale '${name}' was not found.`);\n        return this.scales[name];\n    }\n}\nSideChainScales.scales = {\n    // Wimley-White interfacial hydrophobicity scale\n    'WimleyWhite': {\n        '-': 0,\n        'A': 0.17,\n        'C': -0.24,\n        'D': -0.07,\n        'E': -0.01,\n        'F': -1.13,\n        'G': 0.01,\n        'H': 0.17,\n        'I': -0.31,\n        'K': 0.99,\n        'L': -0.56,\n        'M': -0.23,\n        'N': 0.42,\n        'P': 0.45,\n        'Q': 0.58,\n        'R': 0.81,\n        'S': 0.13,\n        'T': 0.14,\n        'V': 0.07,\n        'W': -1.85,\n        'Y': -0.94,\n    },\n    'categorial': {\n        '-': 0,\n        'A': 1,\n        'C': 2,\n        'D': 3,\n        'E': 4,\n        'F': 5,\n        'G': 6,\n        'H': 7,\n        'I': 8,\n        'K': 9,\n        'L': 10,\n        'M': 11,\n        'N': 12,\n        'P': 13,\n        'Q': 14,\n        'R': 15,\n        'S': 16,\n        'T': 17,\n        'V': 18,\n        'W': 19,\n        'Y': 20,\n    },\n};\n/**\n * Class to categorial encode/decode aligned amino acid residues sequence.\n *\n * @export\n * @class AlignedSequenceEncoder\n */\nexport class AlignedSequenceEncoder {\n    constructor(scale = 'categorial') {\n        this.aa2num = SideChainScales.getScale(scale);\n        this.num2aa = {};\n        Object.entries(this.aa2num).forEach(([k, v]) => (this.num2aa[v] = k));\n    }\n    /**\n       * Truncate NH2 and -COOH terminals of the given sequence.\n       *\n       * @static\n       * @param {string} seq The sequence provided.\n       * @return {string} Truncated sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    static _truncateSequence(seq) {\n        let start = 0;\n        let end = seq.length;\n        const termina = ['NH2', 'COOH'];\n        if (seq.startsWith(termina[0])) {\n            const l = termina[0].length; // Cut only 'NH2' without following '-'.\n            assert(seq[l] == '-', `Wrong sequence format: ${termina[0]} without following '-' in '${seq}'.`);\n            start = l;\n        }\n        if (seq.endsWith(termina[1])) {\n            const l = termina[1].length + 1; // Cut both 'COOH' and precending '-'.\n            assert(seq[end - l] == '-', `Wrong sequence format: ${termina[1]} without '-' precending in '${seq}'.`);\n            end -= l;\n        }\n        return seq.substring(start, end);\n    }\n    /**\n       * Cuts auxiliary defises before a residue.\n       *\n       * @static\n       * @param {string} seq The sequence to process.\n       * @return {string} Processed sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    static _dropDefises(seq) {\n        return seq.replace(/(-)([^-]+)/g, '$2');\n    }\n    /**\n       * Performs truncation and cutting auxiliary defises.\n       *\n       * @static\n       * @param {string} sequence The sequence work under process.\n       * @return {string} Result of cleaning.\n       * @memberof AlignedSequenceEncoder\n       */\n    static clean(sequence) {\n        return AlignedSequenceEncoder._dropDefises(AlignedSequenceEncoder._truncateSequence(sequence));\n    }\n    /**\n       * Categorial encode of the sequence provided.\n       *\n       * @param {string} sequence The sequence.\n       * @return {number[]} Encoded vector.\n       * @memberof AlignedSequenceEncoder\n       */\n    encode(sequence) {\n        const nItems = sequence.length;\n        const values = new Array(nItems).fill(0);\n        for (let i = 0; i < nItems; ++i) {\n            const char = sequence[i];\n            assert(char in this.aa2num, `Unknown char '${char}' found in sequence '${sequence}'`);\n            values[i] = this.encodeLettter(char);\n        }\n        return values;\n    }\n    encodeLettter(letter) {\n        return this.aa2num[letter];\n    }\n    /**\n       * Decode the encoded vector into the sequence back.\n       *\n       * @param {number[]} value The vector encoded.\n       * @return {string} Decoded sequence.\n       * @memberof AlignedSequenceEncoder\n       */\n    decode(value) {\n        let s = '';\n        for (let i = 0; i < value.length; ++i) {\n            const code = value[i];\n            assert(code in this.num2aa, `Unknown code '${code}' found in vector '${value}'`);\n            s += this.num2aa[code];\n        }\n        return s;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sequence-encoder.js","sourceRoot":"","sources":["sequence-encoder.ts"],"names":[],"mappings":"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","sourcesContent":["import {assert} from '@datagrok-libraries/utils/src/vector-operations';\n\ntype SideChainScale = {[name: string]: number};\ntype SideChainScaleCollection = {[name: string]: SideChainScale};\n\nclass SideChainScales {\n  static scales: SideChainScaleCollection = {\n    // Wimley-White interfacial hydrophobicity scale\n    'WimleyWhite': {\n      '-': 0,\n      'A': 0.17,\n      'C': -0.24,\n      'D': -0.07, // Asp-: 1.23\n      'E': -0.01, // Glu-: 2.02\n      'F': -1.13, //\n      'G': 0.01,\n      'H': 0.17, // His+: 0.96\n      'I': -0.31,\n      'K': 0.99, // Lys+\n      'L': -0.56,\n      'M': -0.23,\n      'N': 0.42,\n      'P': 0.45,\n      'Q': 0.58,\n      'R': 0.81, // Arg+\n      'S': 0.13,\n      'T': 0.14,\n      'V': 0.07,\n      'W': -1.85,\n      'Y': -0.94,\n    },\n    'categorial': {\n      '-': 0,\n      'A': 1,\n      'C': 2,\n      'D': 3,\n      'E': 4,\n      'F': 5,\n      'G': 6,\n      'H': 7,\n      'I': 8,\n      'K': 9,\n      'L': 10,\n      'M': 11,\n      'N': 12,\n      'P': 13,\n      'Q': 14,\n      'R': 15,\n      'S': 16,\n      'T': 17,\n      'V': 18,\n      'W': 19,\n      'Y': 20,\n    },\n  };\n\n  static getAvailableScales(): string[] {\n    return Object.entries(this.scales).map(([k, _]) => k);\n  }\n\n  static getScale(name: string): SideChainScale {\n    assert(!(this.scales[name] === undefined), `Scale '${name}' was not found.`);\n    return this.scales[name];\n  }\n}\n\n/**\n * Class to categorial encode/decode aligned amino acid residues sequence.\n *\n * @export\n * @class AlignedSequenceEncoder\n */\nexport class AlignedSequenceEncoder {\n  protected aa2num: SideChainScale;\n  protected num2aa: {[code: number]: string};\n\n  constructor(scale: string = 'categorial') {\n    this.aa2num = SideChainScales.getScale(scale);\n    this.num2aa = {};\n    Object.entries(this.aa2num).forEach(([k, v]) => (this.num2aa[v] = k));\n  }\n\n  /**\n     * Truncate NH2 and -COOH terminals of the given sequence.\n     *\n     * @static\n     * @param {string} seq The sequence provided.\n     * @return {string} Truncated sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  static _truncateSequence(seq: string): string {\n    let start = 0;\n    let end = seq.length;\n    const termina = ['NH2', 'COOH'];\n\n    if (seq.startsWith(termina[0])) {\n      const l = termina[0].length; // Cut only 'NH2' without following '-'.\n      assert(seq[l] == '-', `Wrong sequence format: ${termina[0]} without following '-' in '${seq}'.`);\n      start = l;\n    }\n    if (seq.endsWith(termina[1])) {\n      const l = termina[1].length+1; // Cut both 'COOH' and precending '-'.\n      assert(seq[end-l] == '-', `Wrong sequence format: ${termina[1]} without '-' precending in '${seq}'.`);\n      end -= l;\n    }\n    return seq.substring(start, end);\n  }\n\n  /**\n     * Cuts auxiliary defises before a residue.\n     *\n     * @static\n     * @param {string} seq The sequence to process.\n     * @return {string} Processed sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  static _dropDefises(seq: string): string {\n    return seq.replace(/(-)([^-]+)/g, '$2');\n  }\n\n  /**\n     * Performs truncation and cutting auxiliary defises.\n     *\n     * @static\n     * @param {string} sequence The sequence work under process.\n     * @return {string} Result of cleaning.\n     * @memberof AlignedSequenceEncoder\n     */\n  static clean(sequence: string): string {\n    return AlignedSequenceEncoder._dropDefises(AlignedSequenceEncoder._truncateSequence(sequence));\n  }\n\n  /**\n     * Categorial encode of the sequence provided.\n     *\n     * @param {string} sequence The sequence.\n     * @return {number[]} Encoded vector.\n     * @memberof AlignedSequenceEncoder\n     */\n  public encode(sequence: string): number[] {\n    const nItems = sequence.length;\n    const values = new Array(nItems).fill(0);\n\n    for (let i = 0; i < nItems; ++i) {\n      const char = sequence[i];\n\n      assert(char in this.aa2num, `Unknown char '${char}' found in sequence '${sequence}'`);\n\n      values[i] = this.encodeLettter(char);\n    }\n    return values;\n  }\n\n  public encodeLettter(letter: string): number {\n    return this.aa2num[letter];\n  }\n\n  /**\n     * Decode the encoded vector into the sequence back.\n     *\n     * @param {number[]} value The vector encoded.\n     * @return {string} Decoded sequence.\n     * @memberof AlignedSequenceEncoder\n     */\n  public decode(value: number[]): string {\n    let s: string = '';\n\n    for (let i = 0; i < value.length; ++i) {\n      const code = value[i];\n\n      assert(code in this.num2aa, `Unknown code '${code}' found in vector '${value}'`);\n\n      s += this.num2aa[code];\n    }\n    return s;\n  }\n}\n"]}","export class StringUtils {\n    static hashCode(s) {\n        let hash = 0;\n        if (s.length === 0)\n            return hash;\n        for (let i = 0; i < s.length; i++) {\n            const chr = s.charCodeAt(i);\n            hash = ((hash << 5) - hash) + chr;\n            hash |= 0; // Convert to 32bit integer\n        }\n        return hash;\n    }\n}\nimport { SeqPaletteBase } from './seq-palettes';\nexport class UnknownSeqPalette {\n}\nexport class GrayAllPalette extends UnknownSeqPalette {\n    get(m) {\n        return '#666666';\n    }\n}\nexport class UnknownColorPalette extends UnknownSeqPalette {\n    static buildPalette() {\n        const res = [].concat(...Object.values(SeqPaletteBase.colourPalette));\n        return res;\n    }\n    get(m) {\n        const hash = StringUtils.hashCode(m);\n        const pI = hash % UnknownColorPalette.palette.length;\n        return UnknownColorPalette.palette[pI];\n    }\n}\nUnknownColorPalette.palette = UnknownColorPalette.buildPalette();\nexport class UnknownSeqPalettes extends SeqPaletteBase {\n    static get Gray() {\n        if (this.gray === void 0)\n            this.gray = new GrayAllPalette();\n        return this.gray;\n    }\n    static get Color() {\n        if (this.color === void 0)\n            this.color = new UnknownColorPalette();\n        return this.color;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5rbm93bi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInVua25vd24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBSUEsTUFBTSxPQUFPLFdBQVc7SUFDZixNQUFNLENBQUMsUUFBUSxDQUFDLENBQVM7UUFDOUIsSUFBSSxJQUFJLEdBQVcsQ0FBQyxDQUFDO1FBQ3JCLElBQUksQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDO1lBQ2hCLE9BQU8sSUFBSSxDQUFDO1FBQ2QsS0FBSyxJQUFJLENBQUMsR0FBVyxDQUFDLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUU7WUFDekMsTUFBTSxHQUFHLEdBQVcsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNwQyxJQUFJLEdBQUcsQ0FBQyxDQUFDLElBQUksSUFBSSxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxHQUFHLENBQUM7WUFDbEMsSUFBSSxJQUFJLENBQUMsQ0FBQyxDQUFDLDJCQUEyQjtTQUN2QztRQUNELE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztDQUNGO0FBRUQsT0FBTyxFQUFhLGNBQWMsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRTFELE1BQU0sT0FBZ0IsaUJBQWlCO0NBRXRDO0FBRUQsTUFBTSxPQUFPLGNBQWUsU0FBUSxpQkFBaUI7SUFDNUMsR0FBRyxDQUFDLENBQVM7UUFDbEIsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztDQUNGO0FBR0QsTUFBTSxPQUFPLG1CQUFvQixTQUFRLGlCQUFpQjtJQUdoRCxNQUFNLENBQUMsWUFBWTtRQUN6QixNQUFNLEdBQUcsR0FBSSxFQUFlLENBQUMsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxjQUFjLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQztRQUNwRixPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7SUFFTSxHQUFHLENBQUMsQ0FBUztRQUNsQixNQUFNLElBQUksR0FBVyxXQUFXLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzdDLE1BQU0sRUFBRSxHQUFHLElBQUksR0FBRyxtQkFBbUIsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDO1FBQ3JELE9BQU8sbUJBQW1CLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3pDLENBQUM7O0FBWGEsMkJBQU8sR0FBYSxtQkFBbUIsQ0FBQyxZQUFZLEVBQUUsQ0FBQztBQWN2RSxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsY0FBYztJQUc3QyxNQUFNLEtBQUssSUFBSTtRQUNwQixJQUFJLElBQUksQ0FBQyxJQUFJLEtBQUssS0FBSyxDQUFDO1lBQ3RCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxjQUFjLEVBQUUsQ0FBQztRQUNuQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDbkIsQ0FBQztJQUlNLE1BQU0sS0FBSyxLQUFLO1FBQ3JCLElBQUksSUFBSSxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUM7WUFDdkIsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLG1CQUFtQixFQUFFLENBQUM7UUFDekMsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgdWkgZnJvbSAnZGF0YWdyb2stYXBpL3VpJztcbmltcG9ydCAqIGFzIERHIGZyb20gJ2RhdGFncm9rLWFwaS9kZyc7XG5cbmV4cG9ydCBjbGFzcyBTdHJpbmdVdGlscyB7XG4gIHB1YmxpYyBzdGF0aWMgaGFzaENvZGUoczogc3RyaW5nKTogbnVtYmVyIHtcbiAgICBsZXQgaGFzaDogbnVtYmVyID0gMDtcbiAgICBpZiAocy5sZW5ndGggPT09IDApXG4gICAgICByZXR1cm4gaGFzaDtcbiAgICBmb3IgKGxldCBpOiBudW1iZXIgPSAwOyBpIDwgcy5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgY2hyOiBudW1iZXIgPSBzLmNoYXJDb2RlQXQoaSk7XG4gICAgICBoYXNoID0gKChoYXNoIDw8IDUpIC0gaGFzaCkgKyBjaHI7XG4gICAgICBoYXNoIHw9IDA7IC8vIENvbnZlcnQgdG8gMzJiaXQgaW50ZWdlclxuICAgIH1cbiAgICByZXR1cm4gaGFzaDtcbiAgfVxufVxuXG5pbXBvcnQge1NlcVBhbGV0dGUsIFNlcVBhbGV0dGVCYXNlfSBmcm9tICcuL3NlcS1wYWxldHRlcyc7XG5cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBVbmtub3duU2VxUGFsZXR0ZSBpbXBsZW1lbnRzIFNlcVBhbGV0dGUge1xuICBwdWJsaWMgYWJzdHJhY3QgZ2V0KG06IHN0cmluZyk6IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIEdyYXlBbGxQYWxldHRlIGV4dGVuZHMgVW5rbm93blNlcVBhbGV0dGUge1xuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgcmV0dXJuICcjNjY2NjY2JztcbiAgfVxufVxuXG5cbmV4cG9ydCBjbGFzcyBVbmtub3duQ29sb3JQYWxldHRlIGV4dGVuZHMgVW5rbm93blNlcVBhbGV0dGUge1xuICBwdWJsaWMgc3RhdGljIHBhbGV0dGU6IHN0cmluZ1tdID0gVW5rbm93bkNvbG9yUGFsZXR0ZS5idWlsZFBhbGV0dGUoKTtcblxuICBwcml2YXRlIHN0YXRpYyBidWlsZFBhbGV0dGUoKTogc3RyaW5nW10ge1xuICAgIGNvbnN0IHJlcyA9IChbXSBhcyBzdHJpbmdbXSkuY29uY2F0KC4uLk9iamVjdC52YWx1ZXMoU2VxUGFsZXR0ZUJhc2UuY29sb3VyUGFsZXR0ZSkpO1xuICAgIHJldHVybiByZXM7XG4gIH1cblxuICBwdWJsaWMgZ2V0KG06IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3QgaGFzaDogbnVtYmVyID0gU3RyaW5nVXRpbHMuaGFzaENvZGUobSk7XG4gICAgY29uc3QgcEkgPSBoYXNoICUgVW5rbm93bkNvbG9yUGFsZXR0ZS5wYWxldHRlLmxlbmd0aDtcbiAgICByZXR1cm4gVW5rbm93bkNvbG9yUGFsZXR0ZS5wYWxldHRlW3BJXTtcbiAgfVxufVxuXG5leHBvcnQgY2xhc3MgVW5rbm93blNlcVBhbGV0dGVzIGV4dGVuZHMgU2VxUGFsZXR0ZUJhc2Uge1xuICBwcml2YXRlIHN0YXRpYyBncmF5OiBTZXFQYWxldHRlO1xuXG4gIHB1YmxpYyBzdGF0aWMgZ2V0IEdyYXkoKTogU2VxUGFsZXR0ZSB7XG4gICAgaWYgKHRoaXMuZ3JheSA9PT0gdm9pZCAwKVxuICAgICAgdGhpcy5ncmF5ID0gbmV3IEdyYXlBbGxQYWxldHRlKCk7XG4gICAgcmV0dXJuIHRoaXMuZ3JheTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNvbG9yOiBTZXFQYWxldHRlO1xuXG4gIHB1YmxpYyBzdGF0aWMgZ2V0IENvbG9yKCk6IFNlcVBhbGV0dGUge1xuICAgIGlmICh0aGlzLmNvbG9yID09PSB2b2lkIDApXG4gICAgICB0aGlzLmNvbG9yID0gbmV3IFVua25vd25Db2xvclBhbGV0dGUoKTtcbiAgICByZXR1cm4gdGhpcy5jb2xvcjtcbiAgfVxufVxuIl19","const undefinedColor = 'rgb(100,100,100)';\nconst grayColor = '#808080';\nexport var DrawStyle;\n(function (DrawStyle) {\n    DrawStyle[\"MSA\"] = \"MSA\";\n    DrawStyle[\"classic\"] = \"classic\";\n})(DrawStyle || (DrawStyle = {}));\n/**\n * A function that prints a string aligned to left or centered.\n *\n * @param {number} x x coordinate.\n * @param {number} y y coordinate.\n * @param {number} w Width.\n * @param {number} h Height.\n * @param {CanvasRenderingContext2D} g Canvas rendering context.\n * @param {string} s String to print.\n * @param {string} [color=undefinedColor] String color.\n * @param {number} [pivot=0] Pirvot.\n * @param {boolean} [left=false] Is left aligned.\n * @param {number} [transparencyRate=0.0] Transparency rate where 1.0 is fully transparent\n * @param {string} [separator=''] Is separator for sequence.\n * @param {boolean} [last=false] Is checker if element last or not.\n * @param drawStyle Is draw style. MSA - for multicharSeq, classic - for other seq.\n * @param maxWord Is array of max words for each line.\n * @param maxWordIdx Is index of word we currently draw.\n * @param gridCell Is grid cell, new for updating data in maxWord while rendering.\n * @return {number} x coordinate to start printing at.\n */\nexport function printLeftOrCentered(x, y, w, h, g, s, color = undefinedColor, pivot = 0, left = false, transparencyRate = 1.0, separator = '', last = false, drawStyle = DrawStyle.classic, maxWord = {}, maxWordIdx = 0, gridCell = {}) {\n    var _a;\n    g.textAlign = 'start';\n    const colorPart = s.substring(0);\n    let grayPart = last ? '' : separator;\n    if (drawStyle === DrawStyle.MSA) {\n        grayPart = '';\n    }\n    let textSize = g.measureText(colorPart + grayPart);\n    const indent = 5;\n    let maxColorTextSize = g.measureText(colorPart).width;\n    let colorTextSize = g.measureText(colorPart).width;\n    const dy = (textSize.fontBoundingBoxAscent + textSize.fontBoundingBoxDescent) / 2;\n    textSize = textSize.width;\n    if (drawStyle === DrawStyle.MSA) {\n        maxColorTextSize = maxWord[maxWordIdx];\n        textSize = maxWord[maxWordIdx];\n        if (maxWordIdx > ((_a = maxWord['bio-maxIndex']) !== null && _a !== void 0 ? _a : 0)) {\n            maxWord['bio-maxIndex'] = maxWordIdx;\n            gridCell.cell.column.temp = maxWord;\n        }\n    }\n    function draw(dx1, dx2) {\n        g.fillStyle = color;\n        g.globalAlpha = transparencyRate;\n        if (drawStyle === DrawStyle.classic) {\n            g.fillText(colorPart, x + dx1, y + dy);\n            g.fillStyle = grayColor;\n            g.fillText(grayPart, x + dx2, y + dy);\n        }\n        if (drawStyle === DrawStyle.MSA) {\n            g.fillStyle = color;\n            g.fillText(colorPart, x + dx1 + ((maxWord[maxWordIdx] - colorTextSize) / 2), y + dy);\n        }\n    }\n    if (left || textSize > w) {\n        draw(indent, indent + maxColorTextSize);\n        return x + maxColorTextSize + g.measureText(grayPart).width;\n    }\n    else {\n        const dx = (w - textSize) / 2;\n        draw(dx, dx + maxColorTextSize);\n        return x + dx + maxColorTextSize;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2VsbC1yZW5kZXJlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImNlbGwtcmVuZGVyZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxjQUFjLEdBQUcsa0JBQWtCLENBQUM7QUFDMUMsTUFBTSxTQUFTLEdBQUcsU0FBUyxDQUFDO0FBRTVCLE1BQU0sQ0FBTixJQUFZLFNBR1g7QUFIRCxXQUFZLFNBQVM7SUFDbkIsd0JBQVcsQ0FBQTtJQUNYLGdDQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFIVyxTQUFTLEtBQVQsU0FBUyxRQUdwQjtBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW9CRztBQUNILE1BQU0sVUFBVSxtQkFBbUIsQ0FDakMsQ0FBUyxFQUFFLENBQVMsRUFBRSxDQUFTLEVBQUUsQ0FBUyxFQUMxQyxDQUEyQixFQUFFLENBQVMsRUFBRSxLQUFLLEdBQUcsY0FBYyxFQUM5RCxRQUFnQixDQUFDLEVBQUUsSUFBSSxHQUFHLEtBQUssRUFBRSxtQkFBMkIsR0FBRyxFQUMvRCxZQUFvQixFQUFFLEVBQUUsT0FBZ0IsS0FBSyxFQUFFLFlBQXVCLFNBQVMsQ0FBQyxPQUFPLEVBQUUsVUFBdUMsRUFBRSxFQUFFLGFBQXFCLENBQUMsRUFBRSxXQUFnQixFQUFFOztJQUM5SyxDQUFDLENBQUMsU0FBUyxHQUFHLE9BQU8sQ0FBQztJQUN0QixNQUFNLFNBQVMsR0FBRyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2pDLElBQUksUUFBUSxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7SUFDckMsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLEdBQUcsRUFBRTtRQUMvQixRQUFRLEdBQUcsRUFBRSxDQUFDO0tBQ2Y7SUFFRCxJQUFJLFFBQVEsR0FBUSxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsR0FBRyxRQUFRLENBQUMsQ0FBQztJQUN4RCxNQUFNLE1BQU0sR0FBRyxDQUFDLENBQUM7SUFFakIsSUFBSSxnQkFBZ0IsR0FBRyxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEtBQUssQ0FBQztJQUN0RCxJQUFJLGFBQWEsR0FBRyxDQUFDLENBQUMsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEtBQUssQ0FBQztJQUNuRCxNQUFNLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxxQkFBcUIsR0FBRyxRQUFRLENBQUMsc0JBQXNCLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDbEYsUUFBUSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUM7SUFDMUIsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLEdBQUcsRUFBRTtRQUMvQixnQkFBZ0IsR0FBRyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDdkMsUUFBUSxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUMvQixJQUFJLFVBQVUsR0FBRyxDQUFDLE1BQUEsT0FBTyxDQUFDLGNBQWMsQ0FBQyxtQ0FBSSxDQUFDLENBQUMsRUFBRTtZQUMvQyxPQUFPLENBQUMsY0FBYyxDQUFDLEdBQUcsVUFBVSxDQUFDO1lBQ3JDLFFBQVEsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksR0FBRyxPQUFPLENBQUM7U0FDckM7S0FDRjtJQUVELFNBQVMsSUFBSSxDQUFDLEdBQVcsRUFBRSxHQUFXO1FBQ3BDLENBQUMsQ0FBQyxTQUFTLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLENBQUMsQ0FBQyxXQUFXLEdBQUcsZ0JBQWdCLENBQUM7UUFDakMsSUFBSSxTQUFTLEtBQUssU0FBUyxDQUFDLE9BQU8sRUFBRTtZQUNuQyxDQUFDLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxDQUFDLEdBQUcsR0FBRyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztZQUN2QyxDQUFDLENBQUMsU0FBUyxHQUFHLFNBQVMsQ0FBQztZQUN4QixDQUFDLENBQUMsUUFBUSxDQUFDLFFBQVEsRUFBRSxDQUFDLEdBQUcsR0FBRyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztTQUN2QztRQUNELElBQUksU0FBUyxLQUFLLFNBQVMsQ0FBQyxHQUFHLEVBQUU7WUFDL0IsQ0FBQyxDQUFDLFNBQVMsR0FBRyxLQUFLLENBQUM7WUFDcEIsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxHQUFHLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQztTQUN0RjtJQUNILENBQUM7SUFFRCxJQUFJLElBQUksSUFBSSxRQUFRLEdBQUcsQ0FBQyxFQUFFO1FBQ3hCLElBQUksQ0FBQyxNQUFNLEVBQUUsTUFBTSxHQUFHLGdCQUFnQixDQUFDLENBQUM7UUFDeEMsT0FBTyxDQUFDLEdBQUcsZ0JBQWdCLEdBQUcsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxLQUFLLENBQUM7S0FFN0Q7U0FBTTtRQUNMLE1BQU0sRUFBRSxHQUFHLENBQUMsQ0FBQyxHQUFHLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM5QixJQUFJLENBQUMsRUFBRSxFQUFFLEVBQUUsR0FBRyxnQkFBZ0IsQ0FBQyxDQUFDO1FBQ2hDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsR0FBRyxnQkFBZ0IsQ0FBQztLQUNsQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJjb25zdCB1bmRlZmluZWRDb2xvciA9ICdyZ2IoMTAwLDEwMCwxMDApJztcbmNvbnN0IGdyYXlDb2xvciA9ICcjODA4MDgwJztcblxuZXhwb3J0IGVudW0gRHJhd1N0eWxlIHtcbiAgTVNBID0gJ01TQScsXG4gIGNsYXNzaWMgPSAnY2xhc3NpYycsXG59XG5cbi8qKlxuICogQSBmdW5jdGlvbiB0aGF0IHByaW50cyBhIHN0cmluZyBhbGlnbmVkIHRvIGxlZnQgb3IgY2VudGVyZWQuXG4gKlxuICogQHBhcmFtIHtudW1iZXJ9IHggeCBjb29yZGluYXRlLlxuICogQHBhcmFtIHtudW1iZXJ9IHkgeSBjb29yZGluYXRlLlxuICogQHBhcmFtIHtudW1iZXJ9IHcgV2lkdGguXG4gKiBAcGFyYW0ge251bWJlcn0gaCBIZWlnaHQuXG4gKiBAcGFyYW0ge0NhbnZhc1JlbmRlcmluZ0NvbnRleHQyRH0gZyBDYW52YXMgcmVuZGVyaW5nIGNvbnRleHQuXG4gKiBAcGFyYW0ge3N0cmluZ30gcyBTdHJpbmcgdG8gcHJpbnQuXG4gKiBAcGFyYW0ge3N0cmluZ30gW2NvbG9yPXVuZGVmaW5lZENvbG9yXSBTdHJpbmcgY29sb3IuXG4gKiBAcGFyYW0ge251bWJlcn0gW3Bpdm90PTBdIFBpcnZvdC5cbiAqIEBwYXJhbSB7Ym9vbGVhbn0gW2xlZnQ9ZmFsc2VdIElzIGxlZnQgYWxpZ25lZC5cbiAqIEBwYXJhbSB7bnVtYmVyfSBbdHJhbnNwYXJlbmN5UmF0ZT0wLjBdIFRyYW5zcGFyZW5jeSByYXRlIHdoZXJlIDEuMCBpcyBmdWxseSB0cmFuc3BhcmVudFxuICogQHBhcmFtIHtzdHJpbmd9IFtzZXBhcmF0b3I9JyddIElzIHNlcGFyYXRvciBmb3Igc2VxdWVuY2UuXG4gKiBAcGFyYW0ge2Jvb2xlYW59IFtsYXN0PWZhbHNlXSBJcyBjaGVja2VyIGlmIGVsZW1lbnQgbGFzdCBvciBub3QuXG4gKiBAcGFyYW0gZHJhd1N0eWxlIElzIGRyYXcgc3R5bGUuIE1TQSAtIGZvciBtdWx0aWNoYXJTZXEsIGNsYXNzaWMgLSBmb3Igb3RoZXIgc2VxLlxuICogQHBhcmFtIG1heFdvcmQgSXMgYXJyYXkgb2YgbWF4IHdvcmRzIGZvciBlYWNoIGxpbmUuXG4gKiBAcGFyYW0gbWF4V29yZElkeCBJcyBpbmRleCBvZiB3b3JkIHdlIGN1cnJlbnRseSBkcmF3LlxuICogQHBhcmFtIGdyaWRDZWxsIElzIGdyaWQgY2VsbCwgbmV3IGZvciB1cGRhdGluZyBkYXRhIGluIG1heFdvcmQgd2hpbGUgcmVuZGVyaW5nLlxuICogQHJldHVybiB7bnVtYmVyfSB4IGNvb3JkaW5hdGUgdG8gc3RhcnQgcHJpbnRpbmcgYXQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmludExlZnRPckNlbnRlcmVkKFxuICB4OiBudW1iZXIsIHk6IG51bWJlciwgdzogbnVtYmVyLCBoOiBudW1iZXIsXG4gIGc6IENhbnZhc1JlbmRlcmluZ0NvbnRleHQyRCwgczogc3RyaW5nLCBjb2xvciA9IHVuZGVmaW5lZENvbG9yLFxuICBwaXZvdDogbnVtYmVyID0gMCwgbGVmdCA9IGZhbHNlLCB0cmFuc3BhcmVuY3lSYXRlOiBudW1iZXIgPSAxLjAsXG4gIHNlcGFyYXRvcjogc3RyaW5nID0gJycsIGxhc3Q6IGJvb2xlYW4gPSBmYWxzZSwgZHJhd1N0eWxlOiBEcmF3U3R5bGUgPSBEcmF3U3R5bGUuY2xhc3NpYywgbWF4V29yZDogeyBbaW5kZXg6IHN0cmluZ106IG51bWJlciB9ID0ge30sIG1heFdvcmRJZHg6IG51bWJlciA9IDAsIGdyaWRDZWxsOiBhbnkgPSB7fSk6IG51bWJlciB7XG4gIGcudGV4dEFsaWduID0gJ3N0YXJ0JztcbiAgY29uc3QgY29sb3JQYXJ0ID0gcy5zdWJzdHJpbmcoMCk7XG4gIGxldCBncmF5UGFydCA9IGxhc3QgPyAnJyA6IHNlcGFyYXRvcjtcbiAgaWYgKGRyYXdTdHlsZSA9PT0gRHJhd1N0eWxlLk1TQSkge1xuICAgIGdyYXlQYXJ0ID0gJyc7XG4gIH1cblxuICBsZXQgdGV4dFNpemU6IGFueSA9IGcubWVhc3VyZVRleHQoY29sb3JQYXJ0ICsgZ3JheVBhcnQpO1xuICBjb25zdCBpbmRlbnQgPSA1O1xuXG4gIGxldCBtYXhDb2xvclRleHRTaXplID0gZy5tZWFzdXJlVGV4dChjb2xvclBhcnQpLndpZHRoO1xuICBsZXQgY29sb3JUZXh0U2l6ZSA9IGcubWVhc3VyZVRleHQoY29sb3JQYXJ0KS53aWR0aDtcbiAgY29uc3QgZHkgPSAodGV4dFNpemUuZm9udEJvdW5kaW5nQm94QXNjZW50ICsgdGV4dFNpemUuZm9udEJvdW5kaW5nQm94RGVzY2VudCkgLyAyO1xuICB0ZXh0U2l6ZSA9IHRleHRTaXplLndpZHRoO1xuICBpZiAoZHJhd1N0eWxlID09PSBEcmF3U3R5bGUuTVNBKSB7XG4gICAgbWF4Q29sb3JUZXh0U2l6ZSA9IG1heFdvcmRbbWF4V29yZElkeF07XG4gICAgdGV4dFNpemUgPSBtYXhXb3JkW21heFdvcmRJZHhdO1xuICAgIGlmIChtYXhXb3JkSWR4ID4gKG1heFdvcmRbJ2Jpby1tYXhJbmRleCddID8/IDApKSB7XG4gICAgICBtYXhXb3JkWydiaW8tbWF4SW5kZXgnXSA9IG1heFdvcmRJZHg7XG4gICAgICBncmlkQ2VsbC5jZWxsLmNvbHVtbi50ZW1wID0gbWF4V29yZDtcbiAgICB9XG4gIH1cblxuICBmdW5jdGlvbiBkcmF3KGR4MTogbnVtYmVyLCBkeDI6IG51bWJlcik6IHZvaWQge1xuICAgIGcuZmlsbFN0eWxlID0gY29sb3I7XG4gICAgZy5nbG9iYWxBbHBoYSA9IHRyYW5zcGFyZW5jeVJhdGU7XG4gICAgaWYgKGRyYXdTdHlsZSA9PT0gRHJhd1N0eWxlLmNsYXNzaWMpIHtcbiAgICAgIGcuZmlsbFRleHQoY29sb3JQYXJ0LCB4ICsgZHgxLCB5ICsgZHkpO1xuICAgICAgZy5maWxsU3R5bGUgPSBncmF5Q29sb3I7XG4gICAgICBnLmZpbGxUZXh0KGdyYXlQYXJ0LCB4ICsgZHgyLCB5ICsgZHkpO1xuICAgIH1cbiAgICBpZiAoZHJhd1N0eWxlID09PSBEcmF3U3R5bGUuTVNBKSB7XG4gICAgICBnLmZpbGxTdHlsZSA9IGNvbG9yO1xuICAgICAgZy5maWxsVGV4dChjb2xvclBhcnQsIHggKyBkeDEgKyAoKG1heFdvcmRbbWF4V29yZElkeF0gLSBjb2xvclRleHRTaXplKSAvIDIpLCB5ICsgZHkpO1xuICAgIH1cbiAgfVxuXG4gIGlmIChsZWZ0IHx8IHRleHRTaXplID4gdykge1xuICAgIGRyYXcoaW5kZW50LCBpbmRlbnQgKyBtYXhDb2xvclRleHRTaXplKTtcbiAgICByZXR1cm4geCArIG1heENvbG9yVGV4dFNpemUgKyBnLm1lYXN1cmVUZXh0KGdyYXlQYXJ0KS53aWR0aDtcblxuICB9IGVsc2Uge1xuICAgIGNvbnN0IGR4ID0gKHcgLSB0ZXh0U2l6ZSkgLyAyO1xuICAgIGRyYXcoZHgsIGR4ICsgbWF4Q29sb3JUZXh0U2l6ZSk7XG4gICAgcmV0dXJuIHggKyBkeCArIG1heENvbG9yVGV4dFNpemU7XG4gIH1cbn1cblxuIl19","import * as DG from 'datagrok-api/dg';\nimport { UnitsHandler } from './units-handler';\n/** Class for parsing FASTA files */\nexport class FastaFileHandler {\n    constructor(fileContent) {\n        this._descriptionsArray = []; // parsed FASTA descriptions\n        this._sequencesArray = []; // parsed FASTA sequeces\n        this._fileContent = fileContent;\n        this.parseColumns();\n    }\n    // private _columnsParsed: boolean = false;\n    get descriptionsArray() { return this._descriptionsArray; }\n    get sequencesArray() { return this._sequencesArray; }\n    /**\n     * Helper method to parse a macromolecule from a FASTA file (string)\n     *\n     * @param {number} startOfSequence  index of macromolecule substring beginning\n     * @param {number} endOfSequence  index of macromolecule substring end\n  \n     * @return {string} parsed macromolecule\n     */\n    parseMacromolecule(startOfSequence, endOfSequence) {\n        const seq = this._fileContent.slice(startOfSequence, endOfSequence);\n        const seqArray = seq.split(/\\s/);\n        return seqArray.join('');\n    }\n    /** Parse descriptions and sequences from a FASTA string */\n    parseColumns() {\n        const regex = /^>(.*)$/gm; // match 'description' lines starting with >\n        let startOfSequence = 0;\n        let match; // match.index is the beginning of the matched line\n        while (match = regex.exec(this._fileContent)) {\n            const description = this._fileContent.substring(match.index + 1, regex.lastIndex);\n            this._descriptionsArray.push(description);\n            if (startOfSequence !== 0)\n                this._sequencesArray.push(this.parseMacromolecule(startOfSequence, match.index));\n            startOfSequence = regex.lastIndex + 1;\n        }\n        this._sequencesArray.push(this.parseMacromolecule(startOfSequence, -1));\n        // this._columnsParsed = true;\n    }\n    /**\n     * File-handler method for import as FASTA\n     *\n     * @return {DG.DataFrame[]} dataframe with parsed FASTA content\n     */\n    importFasta() {\n        const descriptionsArrayCol = DG.Column.fromStrings('description', this.descriptionsArray);\n        const sequenceCol = DG.Column.fromStrings('sequence', this.sequencesArray);\n        sequenceCol.semType = DG.SEMTYPE.MACROMOLECULE;\n        // here should go the code from units handler\n        UnitsHandler.setUnitsToFastaColumn(sequenceCol);\n        return [DG.DataFrame.fromColumns([\n                descriptionsArrayCol,\n                sequenceCol,\n            ])];\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFzdGEtaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImZhc3RhLWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0EsT0FBTyxLQUFLLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUV0QyxPQUFPLEVBQUMsWUFBWSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFFN0Msb0NBQW9DO0FBQ3BDLE1BQU0sT0FBTyxnQkFBZ0I7SUFnRTNCLFlBQVksV0FBbUI7UUE5RHZCLHVCQUFrQixHQUFhLEVBQUUsQ0FBQyxDQUFDLDRCQUE0QjtRQUMvRCxvQkFBZSxHQUFhLEVBQUUsQ0FBQyxDQUFDLHdCQUF3QjtRQThEOUQsSUFBSSxDQUFDLFlBQVksR0FBRyxXQUFXLENBQUM7UUFDaEMsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUEvREQsMkNBQTJDO0lBRTNDLElBQVcsaUJBQWlCLEtBQWUsT0FBTyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQyxDQUFDO0lBRTVFLElBQVcsY0FBYyxLQUFlLE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxDQUFDLENBQUM7SUFFdEU7Ozs7Ozs7T0FPRztJQUNLLGtCQUFrQixDQUN4QixlQUF1QixFQUN2QixhQUFxQjtRQUVyQixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxlQUFlLEVBQUUsYUFBYSxDQUFDLENBQUM7UUFDcEUsTUFBTSxRQUFRLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNqQyxPQUFPLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDM0IsQ0FBQztJQUVELDJEQUEyRDtJQUNuRCxZQUFZO1FBQ2xCLE1BQU0sS0FBSyxHQUFHLFdBQVcsQ0FBQyxDQUFDLDRDQUE0QztRQUV2RSxJQUFJLGVBQWUsR0FBRyxDQUFDLENBQUM7UUFDeEIsSUFBSSxLQUFLLENBQUMsQ0FBQyxtREFBbUQ7UUFDOUQsT0FBTyxLQUFLLEdBQUcsS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUU7WUFDNUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEtBQUssR0FBRyxDQUFDLEVBQUUsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ2xGLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7WUFDMUMsSUFBSSxlQUFlLEtBQUssQ0FBQztnQkFDdkIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGtCQUFrQixDQUFDLGVBQWUsRUFBRSxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztZQUNuRixlQUFlLEdBQUcsS0FBSyxDQUFDLFNBQVMsR0FBRyxDQUFDLENBQUM7U0FDdkM7UUFDRCxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsZUFBZSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUV4RSw4QkFBOEI7SUFDaEMsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxXQUFXO1FBQ2hCLE1BQU0sb0JBQW9CLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUMsYUFBYSxFQUFFLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQzFGLE1BQU0sV0FBVyxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUM7UUFDM0UsV0FBVyxDQUFDLE9BQU8sR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQztRQUUvQyw2Q0FBNkM7UUFDN0MsWUFBWSxDQUFDLHFCQUFxQixDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRWhELE9BQU8sQ0FBQyxFQUFFLENBQUMsU0FBUyxDQUFDLFdBQVcsQ0FBQztnQkFDL0Isb0JBQW9CO2dCQUNwQixXQUFXO2FBQ1osQ0FBQyxDQUFDLENBQUM7SUFDTixDQUFDO0NBTUYiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBEbyBub3QgY2hhbmdlIHRoZXNlIGltcG9ydCBsaW5lcyB0byBtYXRjaCBleHRlcm5hbCBtb2R1bGVzIGluIHdlYnBhY2sgY29uZmlndXJhdGlvbiAqL1xuaW1wb3J0ICogYXMgZ3JvayBmcm9tICdkYXRhZ3Jvay1hcGkvZ3Jvayc7XG5pbXBvcnQgKiBhcyB1aSBmcm9tICdkYXRhZ3Jvay1hcGkvdWknO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcblxuaW1wb3J0IHtVbml0c0hhbmRsZXJ9IGZyb20gJy4vdW5pdHMtaGFuZGxlcic7XG5cbi8qKiBDbGFzcyBmb3IgcGFyc2luZyBGQVNUQSBmaWxlcyAqL1xuZXhwb3J0IGNsYXNzIEZhc3RhRmlsZUhhbmRsZXIge1xuICBwcml2YXRlIF9maWxlQ29udGVudDogc3RyaW5nO1xuICBwcml2YXRlIF9kZXNjcmlwdGlvbnNBcnJheTogc3RyaW5nW10gPSBbXTsgLy8gcGFyc2VkIEZBU1RBIGRlc2NyaXB0aW9uc1xuICBwcml2YXRlIF9zZXF1ZW5jZXNBcnJheTogc3RyaW5nW10gPSBbXTsgLy8gcGFyc2VkIEZBU1RBIHNlcXVlY2VzXG4gIC8vIHByaXZhdGUgX2NvbHVtbnNQYXJzZWQ6IGJvb2xlYW4gPSBmYWxzZTtcblxuICBwdWJsaWMgZ2V0IGRlc2NyaXB0aW9uc0FycmF5KCk6IHN0cmluZ1tdIHsgcmV0dXJuIHRoaXMuX2Rlc2NyaXB0aW9uc0FycmF5OyB9XG5cbiAgcHVibGljIGdldCBzZXF1ZW5jZXNBcnJheSgpOiBzdHJpbmdbXSB7IHJldHVybiB0aGlzLl9zZXF1ZW5jZXNBcnJheTsgfVxuXG4gIC8qKlxuICAgKiBIZWxwZXIgbWV0aG9kIHRvIHBhcnNlIGEgbWFjcm9tb2xlY3VsZSBmcm9tIGEgRkFTVEEgZmlsZSAoc3RyaW5nKVxuICAgKlxuICAgKiBAcGFyYW0ge251bWJlcn0gc3RhcnRPZlNlcXVlbmNlICBpbmRleCBvZiBtYWNyb21vbGVjdWxlIHN1YnN0cmluZyBiZWdpbm5pbmdcbiAgICogQHBhcmFtIHtudW1iZXJ9IGVuZE9mU2VxdWVuY2UgIGluZGV4IG9mIG1hY3JvbW9sZWN1bGUgc3Vic3RyaW5nIGVuZFxuXG4gICAqIEByZXR1cm4ge3N0cmluZ30gcGFyc2VkIG1hY3JvbW9sZWN1bGVcbiAgICovXG4gIHByaXZhdGUgcGFyc2VNYWNyb21vbGVjdWxlKFxuICAgIHN0YXJ0T2ZTZXF1ZW5jZTogbnVtYmVyLFxuICAgIGVuZE9mU2VxdWVuY2U6IG51bWJlclxuICApOiBzdHJpbmcge1xuICAgIGNvbnN0IHNlcSA9IHRoaXMuX2ZpbGVDb250ZW50LnNsaWNlKHN0YXJ0T2ZTZXF1ZW5jZSwgZW5kT2ZTZXF1ZW5jZSk7XG4gICAgY29uc3Qgc2VxQXJyYXkgPSBzZXEuc3BsaXQoL1xccy8pO1xuICAgIHJldHVybiBzZXFBcnJheS5qb2luKCcnKTtcbiAgfVxuXG4gIC8qKiBQYXJzZSBkZXNjcmlwdGlvbnMgYW5kIHNlcXVlbmNlcyBmcm9tIGEgRkFTVEEgc3RyaW5nICovXG4gIHByaXZhdGUgcGFyc2VDb2x1bW5zKCkge1xuICAgIGNvbnN0IHJlZ2V4ID0gL14+KC4qKSQvZ207IC8vIG1hdGNoICdkZXNjcmlwdGlvbicgbGluZXMgc3RhcnRpbmcgd2l0aCA+XG5cbiAgICBsZXQgc3RhcnRPZlNlcXVlbmNlID0gMDtcbiAgICBsZXQgbWF0Y2g7IC8vIG1hdGNoLmluZGV4IGlzIHRoZSBiZWdpbm5pbmcgb2YgdGhlIG1hdGNoZWQgbGluZVxuICAgIHdoaWxlIChtYXRjaCA9IHJlZ2V4LmV4ZWModGhpcy5fZmlsZUNvbnRlbnQpKSB7XG4gICAgICBjb25zdCBkZXNjcmlwdGlvbiA9IHRoaXMuX2ZpbGVDb250ZW50LnN1YnN0cmluZyhtYXRjaC5pbmRleCArIDEsIHJlZ2V4Lmxhc3RJbmRleCk7XG4gICAgICB0aGlzLl9kZXNjcmlwdGlvbnNBcnJheS5wdXNoKGRlc2NyaXB0aW9uKTtcbiAgICAgIGlmIChzdGFydE9mU2VxdWVuY2UgIT09IDApXG4gICAgICAgIHRoaXMuX3NlcXVlbmNlc0FycmF5LnB1c2godGhpcy5wYXJzZU1hY3JvbW9sZWN1bGUoc3RhcnRPZlNlcXVlbmNlLCBtYXRjaC5pbmRleCkpO1xuICAgICAgc3RhcnRPZlNlcXVlbmNlID0gcmVnZXgubGFzdEluZGV4ICsgMTtcbiAgICB9XG4gICAgdGhpcy5fc2VxdWVuY2VzQXJyYXkucHVzaCh0aGlzLnBhcnNlTWFjcm9tb2xlY3VsZShzdGFydE9mU2VxdWVuY2UsIC0xKSk7XG5cbiAgICAvLyB0aGlzLl9jb2x1bW5zUGFyc2VkID0gdHJ1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBGaWxlLWhhbmRsZXIgbWV0aG9kIGZvciBpbXBvcnQgYXMgRkFTVEFcbiAgICpcbiAgICogQHJldHVybiB7REcuRGF0YUZyYW1lW119IGRhdGFmcmFtZSB3aXRoIHBhcnNlZCBGQVNUQSBjb250ZW50XG4gICAqL1xuICBwdWJsaWMgaW1wb3J0RmFzdGEoKTogREcuRGF0YUZyYW1lIFtdIHtcbiAgICBjb25zdCBkZXNjcmlwdGlvbnNBcnJheUNvbCA9IERHLkNvbHVtbi5mcm9tU3RyaW5ncygnZGVzY3JpcHRpb24nLCB0aGlzLmRlc2NyaXB0aW9uc0FycmF5KTtcbiAgICBjb25zdCBzZXF1ZW5jZUNvbCA9IERHLkNvbHVtbi5mcm9tU3RyaW5ncygnc2VxdWVuY2UnLCB0aGlzLnNlcXVlbmNlc0FycmF5KTtcbiAgICBzZXF1ZW5jZUNvbC5zZW1UeXBlID0gREcuU0VNVFlQRS5NQUNST01PTEVDVUxFO1xuXG4gICAgLy8gaGVyZSBzaG91bGQgZ28gdGhlIGNvZGUgZnJvbSB1bml0cyBoYW5kbGVyXG4gICAgVW5pdHNIYW5kbGVyLnNldFVuaXRzVG9GYXN0YUNvbHVtbihzZXF1ZW5jZUNvbCk7XG5cbiAgICByZXR1cm4gW0RHLkRhdGFGcmFtZS5mcm9tQ29sdW1ucyhbXG4gICAgICBkZXNjcmlwdGlvbnNBcnJheUNvbCxcbiAgICAgIHNlcXVlbmNlQ29sLFxuICAgIF0pXTtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKGZpbGVDb250ZW50OiBzdHJpbmcpIHtcbiAgICB0aGlzLl9maWxlQ29udGVudCA9IGZpbGVDb250ZW50O1xuICAgIHRoaXMucGFyc2VDb2x1bW5zKCk7XG4gIH1cbn1cbiJdfQ==","import { WebLogo } from '../viewers/web-logo';\nimport { UnitsHandler } from './units-handler';\n/** Class for handling conversion of notation systems in Macromolecule columns */\nexport class NotationConverter extends UnitsHandler {\n    constructor(col) {\n        super(col);\n        this._splitter = null;\n    }\n    get splitter() {\n        if (this._splitter === null)\n            this._splitter = WebLogo.getSplitterForColumn(this.column);\n        return this._splitter;\n    }\n    ;\n    toFasta(targetNotation) { return targetNotation === \"fasta\" /* NOTATION.FASTA */; }\n    toSeparator(targetNotation) { return targetNotation === \"separator\" /* NOTATION.SEPARATOR */; }\n    toHelm(targetNotation) { return targetNotation === \"helm\" /* NOTATION.HELM */; }\n    /**\n     * Convert a Macromolecule column from FASTA to SEPARATOR notation\n     *\n     * @param {string} separator  A specific separator to be used\n     * @param {string} fastaGapSymbol  Gap symbol in FASTA, '-' by default\n     * @return {DG.Column}        A new column in SEPARATOR notation\n     */\n    convertFastaToSeparator(separator, fastaGapSymbol = null) {\n        if (fastaGapSymbol === null)\n            fastaGapSymbol = this.defaultGapSymbol;\n        const newColumn = this.getNewColumn(\"separator\" /* NOTATION.SEPARATOR */);\n        // assign the values to the newly created empty column\n        newColumn.init((idx) => {\n            const fastaPolymer = this.column.get(idx);\n            const fastaMonomersArray = this.splitter(fastaPolymer);\n            for (let i = 0; i < fastaMonomersArray.length; i++) {\n                if (fastaMonomersArray[i] === fastaGapSymbol)\n                    fastaMonomersArray[i] = UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n            }\n            return fastaMonomersArray.join(separator);\n        });\n        newColumn.setTag(UnitsHandler.TAGS.separator, separator);\n        return newColumn;\n    }\n    /**\n     * Get the wrapper strings for HELM, depending on the type of the\n     * macromolecule (peptide, DNA, RNA)\n     *\n     * @return {string[]} Array of wrappers\n     */\n    getHelmWrappers() {\n        const prefix = (this.isDna()) ? 'DNA1{' :\n            (this.isRna()) ? 'RNA1{' :\n                (this.isPeptide()) ? 'PEPTIDE1{' :\n                    'Unknown'; // this case should be handled as exceptional\n        if (prefix === 'Unknown')\n            throw new Error('Neither peptide, nor nucleotide');\n        const postfix = '}$$$';\n        const leftWrapper = (this.isDna()) ? 'D(' :\n            (this.isRna()) ? 'R(' : ''; // no wrapper for peptides\n        const rightWrapper = (this.isDna() || this.isRna()) ? ')P' : ''; // no wrapper for peptides\n        return [prefix, leftWrapper, rightWrapper, postfix];\n    }\n    // A helper function for converting strings to HELM\n    convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix) {\n        const monomerArray = this.splitter(sourcePolymer);\n        const monomerHelmArray = monomerArray.map((mm) => {\n            if (mm === sourceGapSymbol)\n                return UnitsHandler._defaultGapSymbolsDict.HELM;\n            else\n                return `${leftWrapper}${mm}${rightWrapper}`;\n        });\n        return `${prefix}${monomerHelmArray.join('.')}${postfix}`;\n    }\n    /**\n     * Convert a string with SEPARATOR/FASTA notation to HELM\n     *\n     * @param {string} sourcePolymer  A string to be converted\n     * @param {string | null} sourceGapSymbol  An optional gap symbol, set to\n     * default values ('-' for FASTA and '' for SEPARATOR) unless specified\n     * @return {string}  The target HELM string\n     */\n    convertStringToHelm(sourcePolymer, sourceGapSymbol = null) {\n        if (sourceGapSymbol === null)\n            sourceGapSymbol = this.defaultGapSymbol;\n        const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n        return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n    }\n    /**\n     * Convert a column to HELM\n     *\n     * @param {string | null} sourceGapSymbol\n     * @return {DG.Column}\n     */\n    convertToHelm(sourceGapSymbol = null) {\n        if (sourceGapSymbol === null)\n            sourceGapSymbol = this.defaultGapSymbol;\n        const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n        const newColumn = this.getNewColumn(\"helm\" /* NOTATION.HELM */);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const sourcePolymer = this.column.get(idx);\n            return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n        });\n        return newColumn;\n    }\n    /**\n     * Convert SEPARATOR column to FASTA notation\n     *\n     * @param {string | null} fastaGapSymbol Optional gap symbol for FASTA\n     * @return {DG.Column}  Converted column\n     */\n    convertSeparatorToFasta(fastaGapSymbol = null) {\n        if (fastaGapSymbol === null)\n            fastaGapSymbol = UnitsHandler._defaultGapSymbolsDict.FASTA;\n        const newColumn = this.getNewColumn(\"fasta\" /* NOTATION.FASTA */);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const separatorPolymer = this.column.get(idx);\n            // items can be monomers or separators\n            const separatorItemsArray = this.splitter(separatorPolymer);\n            const fastaMonomersArray = [];\n            for (let i = 0; i < separatorItemsArray.length; i++) {\n                const item = separatorItemsArray[i];\n                if (item.length === 0) {\n                    fastaMonomersArray.push(fastaGapSymbol);\n                }\n                else if (item.length > 1) {\n                    // the case of a multi-character monomer\n                    const monomer = '[' + item + ']';\n                    fastaMonomersArray.push(monomer);\n                }\n                else {\n                    fastaMonomersArray.push(item);\n                }\n            }\n            return fastaMonomersArray.join('');\n        });\n        return newColumn;\n    }\n    /**\n     *  Convert HELM column to FASTA/SEPARATOR\n     *\n     * @param {string} tgtNotation    Target notation: FASTA or SEPARATOR\n     * @param {string} tgtSeparator   Optional target separator (for HELM ->\n     * @param {string | null} tgtGapSymbol   Optional target gap symbol\n     * SEPARATOR)\n     * @return {DG.Column} Converted column\n     */\n    convertHelm(tgtNotation, tgtSeparator = '', tgtGapSymbol = null) {\n        // This function must not contain calls of isDna() and isRna(), for\n        // source helm columns may contain RNA, DNA and PT across different rows\n        if (tgtGapSymbol === null) {\n            tgtGapSymbol = (this.toFasta(tgtNotation)) ?\n                UnitsHandler._defaultGapSymbolsDict.FASTA :\n                UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n        }\n        if (this.toSeparator(tgtNotation) && tgtSeparator === '')\n            tgtSeparator = this.separator;\n        const helmWrappersRe = /(R\\(|D\\(|\\)|P)/g;\n        const newColumn = this.getNewColumn(tgtNotation);\n        // assign the values to the empty column\n        newColumn.init((idx) => {\n            const helmPolymer = this.column.get(idx);\n            // we cannot use isDna() or isRna() because source helm columns can\n            // contain DNA, RNA and PT in different cells, so the corresponding\n            // tags cannot be set for the whole column\n            const isNucleotide = helmPolymer.startsWith('DNA') || helmPolymer.startsWith('RNA');\n            // items can be monomers or helms\n            const helmItemsArray = this.splitter(helmPolymer);\n            const tgtMonomersArray = [];\n            for (let i = 0; i < helmItemsArray.length; i++) {\n                let item = helmItemsArray[i];\n                if (isNucleotide)\n                    item = item.replace(helmWrappersRe, '');\n                if (item === UnitsHandler._defaultGapSymbolsDict.HELM) {\n                    tgtMonomersArray.push(tgtGapSymbol);\n                }\n                else if (this.toFasta(tgtNotation) && item.length > 1) {\n                    // the case of a multi-character monomer converted to FASTA\n                    const monomer = '[' + item + ']';\n                    tgtMonomersArray.push(monomer);\n                }\n                else {\n                    tgtMonomersArray.push(item);\n                }\n            }\n            return tgtMonomersArray.join(tgtSeparator);\n        });\n        return newColumn;\n    }\n    convertHelmToSeparator() {\n        // TODO: implementatioreturn this.getNewColumn();\n        return this.getNewColumn(\"separator\" /* NOTATION.SEPARATOR */);\n    }\n    /** Dispatcher method for notation conversion\n     *\n     * @param {NOTATION} tgtNotation   Notation we want to convert to\n     * @param {string | null} tgtSeparator   Possible separator\n     * @return {DG.Column}                Converted column\n     */\n    convert(tgtNotation, tgtSeparator = null) {\n        // possible exceptions\n        if (this.notation === tgtNotation)\n            throw new Error('tgt notation is invalid');\n        if (this.toSeparator(tgtNotation) && tgtSeparator === null)\n            throw new Error('tgt separator is not specified');\n        if (this.isFasta() && this.toSeparator(tgtNotation) && tgtSeparator !== null)\n            return this.convertFastaToSeparator(tgtSeparator);\n        else if ((this.isFasta() || this.isSeparator()) && this.toHelm(tgtNotation))\n            return this.convertToHelm();\n        else if (this.isSeparator() && this.toFasta(tgtNotation))\n            return this.convertSeparatorToFasta();\n        else if (this.isHelm() && this.toFasta(tgtNotation)) // the case of HELM\n            return this.convertHelm(tgtNotation);\n        else // this.isHelm() && this.toSeparator(tgtNotation)\n            return this.convertHelm(tgtNotation, tgtSeparator);\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notation-converter.js","sourceRoot":"","sources":["notation-converter.ts"],"names":[],"mappings":"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","sourcesContent":["/* Do not change these import lines to match external modules in webpack configuration */\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport {SplitterFunc, WebLogo} from '../viewers/web-logo';\nimport {UnitsHandler, NOTATION} from './units-handler';\n\n/** Class for handling conversion of notation systems in Macromolecule columns */\nexport class NotationConverter extends UnitsHandler {\n  private _splitter: SplitterFunc | null = null;\n  protected get splitter(): SplitterFunc {\n    if (this._splitter === null)\n      this._splitter = WebLogo.getSplitterForColumn(this.column);\n    return this._splitter;\n  };\n\n  public toFasta(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.FASTA; }\n\n  public toSeparator(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.SEPARATOR; }\n\n  public toHelm(targetNotation: NOTATION): boolean { return targetNotation === NOTATION.HELM; }\n\n  /**\n   * Convert a Macromolecule column from FASTA to SEPARATOR notation\n   *\n   * @param {string} separator  A specific separator to be used\n   * @param {string} fastaGapSymbol  Gap symbol in FASTA, '-' by default\n   * @return {DG.Column}        A new column in SEPARATOR notation\n   */\n  private convertFastaToSeparator(separator: string, fastaGapSymbol: string | null = null): DG.Column {\n    if (fastaGapSymbol === null)\n      fastaGapSymbol = this.defaultGapSymbol;\n\n    const newColumn = this.getNewColumn(NOTATION.SEPARATOR);\n    // assign the values to the newly created empty column\n    newColumn.init((idx: number) => {\n      const fastaPolymer = this.column.get(idx);\n      const fastaMonomersArray = this.splitter(fastaPolymer);\n      for (let i = 0; i < fastaMonomersArray.length; i++) {\n        if (fastaMonomersArray[i] === fastaGapSymbol)\n          fastaMonomersArray[i] = UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n      }\n      return fastaMonomersArray.join(separator);\n    });\n    newColumn.setTag(UnitsHandler.TAGS.separator, separator);\n    return newColumn;\n  }\n\n  /**\n   * Get the wrapper strings for HELM, depending on the type of the\n   * macromolecule (peptide, DNA, RNA)\n   *\n   * @return {string[]} Array of wrappers\n   */\n  private getHelmWrappers(): string[] {\n    const prefix = (this.isDna()) ? 'DNA1{' :\n      (this.isRna()) ? 'RNA1{' :\n        (this.isPeptide()) ? 'PEPTIDE1{' :\n          'Unknown'; // this case should be handled as exceptional\n\n    if (prefix === 'Unknown')\n      throw new Error('Neither peptide, nor nucleotide');\n\n    const postfix = '}$$$';\n    const leftWrapper = (this.isDna()) ? 'D(' :\n      (this.isRna()) ? 'R(' : ''; // no wrapper for peptides\n    const rightWrapper = (this.isDna() || this.isRna()) ? ')P' : ''; // no wrapper for peptides\n    return [prefix, leftWrapper, rightWrapper, postfix];\n  }\n\n  // A helper function for converting strings to HELM\n  private convertToHelmHelper(\n    sourcePolymer: string,\n    sourceGapSymbol: string,\n    prefix: string,\n    leftWrapper: string,\n    rightWrapper: string,\n    postfix: string\n  ): string {\n    const monomerArray = this.splitter(sourcePolymer);\n    const monomerHelmArray: string[] = monomerArray.map((mm: string) => {\n      if (mm === sourceGapSymbol)\n        return UnitsHandler._defaultGapSymbolsDict.HELM;\n      else\n        return `${leftWrapper}${mm}${rightWrapper}`;\n    });\n    return `${prefix}${monomerHelmArray.join('.')}${postfix}`;\n  }\n\n  /**\n   * Convert a string with SEPARATOR/FASTA notation to HELM\n   *\n   * @param {string} sourcePolymer  A string to be converted\n   * @param {string | null} sourceGapSymbol  An optional gap symbol, set to\n   * default values ('-' for FASTA and '' for SEPARATOR) unless specified\n   * @return {string}  The target HELM string\n   */\n  public convertStringToHelm(\n    sourcePolymer: string,\n    sourceGapSymbol: string | null = null\n  ): string {\n    if (sourceGapSymbol === null)\n      sourceGapSymbol = this.defaultGapSymbol;\n    const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n    return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol, prefix, leftWrapper, rightWrapper, postfix);\n  }\n\n  /**\n   * Convert a column to HELM\n   *\n   * @param {string | null} sourceGapSymbol\n   * @return {DG.Column}\n   */\n  private convertToHelm(sourceGapSymbol: string | null = null): DG.Column {\n    if (sourceGapSymbol === null)\n      sourceGapSymbol = this.defaultGapSymbol;\n\n    const [prefix, leftWrapper, rightWrapper, postfix] = this.getHelmWrappers();\n\n    const newColumn = this.getNewColumn(NOTATION.HELM);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const sourcePolymer = this.column.get(idx);\n      return this.convertToHelmHelper(sourcePolymer, sourceGapSymbol!, prefix, leftWrapper, rightWrapper, postfix);\n    });\n    return newColumn;\n  }\n\n  /**\n   * Convert SEPARATOR column to FASTA notation\n   *\n   * @param {string | null} fastaGapSymbol Optional gap symbol for FASTA\n   * @return {DG.Column}  Converted column\n   */\n  private convertSeparatorToFasta(fastaGapSymbol: string | null = null): DG.Column {\n    if (fastaGapSymbol === null)\n      fastaGapSymbol = UnitsHandler._defaultGapSymbolsDict.FASTA;\n\n    const newColumn = this.getNewColumn(NOTATION.FASTA);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const separatorPolymer = this.column.get(idx);\n      // items can be monomers or separators\n      const separatorItemsArray = this.splitter(separatorPolymer);\n      const fastaMonomersArray: string[] = [];\n      for (let i = 0; i < separatorItemsArray.length; i++) {\n        const item = separatorItemsArray[i];\n        if (item.length === 0) {\n          fastaMonomersArray.push(fastaGapSymbol!);\n        } else if (item.length > 1) {\n          // the case of a multi-character monomer\n          const monomer = '[' + item + ']';\n          fastaMonomersArray.push(monomer);\n        } else {\n          fastaMonomersArray.push(item);\n        }\n      }\n      return fastaMonomersArray.join('');\n    });\n    return newColumn;\n  }\n\n  /**\n   *  Convert HELM column to FASTA/SEPARATOR\n   *\n   * @param {string} tgtNotation    Target notation: FASTA or SEPARATOR\n   * @param {string} tgtSeparator   Optional target separator (for HELM ->\n   * @param {string | null} tgtGapSymbol   Optional target gap symbol\n   * SEPARATOR)\n   * @return {DG.Column} Converted column\n   */\n  private convertHelm(\n    tgtNotation: string,\n    tgtSeparator: string = '',\n    tgtGapSymbol: string | null = null\n  ): DG.Column {\n    // This function must not contain calls of isDna() and isRna(), for\n    // source helm columns may contain RNA, DNA and PT across different rows\n    if (tgtGapSymbol === null) {\n      tgtGapSymbol = (this.toFasta(tgtNotation as NOTATION)) ?\n        UnitsHandler._defaultGapSymbolsDict.FASTA :\n        UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n    }\n\n    if (this.toSeparator(tgtNotation as NOTATION) && tgtSeparator === '')\n      tgtSeparator = this.separator;\n\n    const helmWrappersRe = /(R\\(|D\\(|\\)|P)/g;\n    const newColumn = this.getNewColumn(tgtNotation as NOTATION);\n    // assign the values to the empty column\n    newColumn.init((idx: number) => {\n      const helmPolymer = this.column.get(idx);\n\n      // we cannot use isDna() or isRna() because source helm columns can\n      // contain DNA, RNA and PT in different cells, so the corresponding\n      // tags cannot be set for the whole column\n      const isNucleotide = helmPolymer.startsWith('DNA') || helmPolymer.startsWith('RNA');\n\n      // items can be monomers or helms\n      const helmItemsArray = this.splitter(helmPolymer);\n      const tgtMonomersArray: string[] = [];\n      for (let i = 0; i < helmItemsArray.length; i++) {\n        let item = helmItemsArray[i];\n        if (isNucleotide)\n          item = item.replace(helmWrappersRe, '');\n        if (item === UnitsHandler._defaultGapSymbolsDict.HELM) {\n          tgtMonomersArray.push(tgtGapSymbol!);\n        } else if (this.toFasta(tgtNotation as NOTATION) && item.length > 1) {\n          // the case of a multi-character monomer converted to FASTA\n          const monomer = '[' + item + ']';\n          tgtMonomersArray.push(monomer);\n        } else {\n          tgtMonomersArray.push(item);\n        }\n      }\n      return tgtMonomersArray.join(tgtSeparator);\n    });\n    return newColumn;\n  }\n\n  private convertHelmToSeparator(): DG.Column {\n    // TODO: implementatioreturn this.getNewColumn();\n    return this.getNewColumn(NOTATION.SEPARATOR);\n  }\n\n  /** Dispatcher method for notation conversion\n   *\n   * @param {NOTATION} tgtNotation   Notation we want to convert to\n   * @param {string | null} tgtSeparator   Possible separator\n   * @return {DG.Column}                Converted column\n   */\n  public convert(tgtNotation: NOTATION, tgtSeparator: string | null = null): DG.Column {\n    // possible exceptions\n    if (this.notation === tgtNotation)\n      throw new Error('tgt notation is invalid');\n    if (this.toSeparator(tgtNotation) && tgtSeparator === null)\n      throw new Error('tgt separator is not specified');\n\n    if (this.isFasta() && this.toSeparator(tgtNotation) && tgtSeparator !== null)\n      return this.convertFastaToSeparator(tgtSeparator);\n    else if ((this.isFasta() || this.isSeparator()) && this.toHelm(tgtNotation))\n      return this.convertToHelm();\n    else if (this.isSeparator() && this.toFasta(tgtNotation))\n      return this.convertSeparatorToFasta();\n    else if (this.isHelm() && this.toFasta(tgtNotation)) // the case of HELM\n      return this.convertHelm(tgtNotation);\n    else // this.isHelm() && this.toSeparator(tgtNotation)\n      return this.convertHelm(tgtNotation, tgtSeparator!);\n  }\n\n  public constructor(col: DG.Column) {\n    super(col);\n  }\n}\n"]}","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '../viewers/web-logo';\nexport function splitAlignedSequences(peptideColumn) {\n    const splitter = WebLogo.getSplitterForColumn(peptideColumn);\n    const colLen = peptideColumn.length;\n    const resultDf = DG.DataFrame.create(colLen);\n    let monomerList = splitter(peptideColumn.get(0));\n    const columnList = [];\n    // create columns and fill the first row for faster values filling in the next loop\n    for (let i = 0; i < monomerList.length; i++) {\n        const col = resultDf.columns.addNewString((i + 1).toString());\n        col.set(0, monomerList[i] || '-', false);\n        columnList.push(col);\n    }\n    for (let rowIndex = 1; rowIndex < colLen; rowIndex++) {\n        monomerList = splitter(peptideColumn.get(rowIndex));\n        monomerList.forEach((monomer, colIndex) => columnList[colIndex].set(rowIndex, monomer || '-', false));\n    }\n    return resultDf;\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BsaXR0ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzcGxpdHRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRXRDLE9BQU8sRUFBQyxPQUFPLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUU1QyxNQUFNLFVBQVUscUJBQXFCLENBQUMsYUFBZ0M7SUFDcEUsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLG9CQUFvQixDQUFDLGFBQWEsQ0FBQyxDQUFDO0lBQzdELE1BQU0sTUFBTSxHQUFHLGFBQWEsQ0FBQyxNQUFNLENBQUM7SUFDcEMsTUFBTSxRQUFRLEdBQUcsRUFBRSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDN0MsSUFBSSxXQUFXLEdBQUcsUUFBUSxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFFLENBQUMsQ0FBQztJQUNsRCxNQUFNLFVBQVUsR0FBd0IsRUFBRSxDQUFDO0lBRTNDLG1GQUFtRjtJQUNuRixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsV0FBVyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtRQUMzQyxNQUFNLEdBQUcsR0FBRyxRQUFRLENBQUMsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDO1FBQzlELEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsSUFBSSxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDekMsVUFBVSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztLQUN0QjtJQUVELEtBQUssSUFBSSxRQUFRLEdBQUcsQ0FBQyxFQUFFLFFBQVEsR0FBRyxNQUFNLEVBQUUsUUFBUSxFQUFFLEVBQUU7UUFDcEQsV0FBVyxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBRSxDQUFDLENBQUM7UUFDckQsV0FBVyxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsRUFBRSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxHQUFHLENBQUMsUUFBUSxFQUFFLE9BQU8sSUFBSSxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztLQUN2RztJQUVELE9BQU8sUUFBUSxDQUFDO0FBQ2xCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG5pbXBvcnQge1dlYkxvZ299IGZyb20gJy4uL3ZpZXdlcnMvd2ViLWxvZ28nO1xuXG5leHBvcnQgZnVuY3Rpb24gc3BsaXRBbGlnbmVkU2VxdWVuY2VzKHBlcHRpZGVDb2x1bW46IERHLkNvbHVtbjxzdHJpbmc+KTogREcuRGF0YUZyYW1lIHtcbiAgY29uc3Qgc3BsaXR0ZXIgPSBXZWJMb2dvLmdldFNwbGl0dGVyRm9yQ29sdW1uKHBlcHRpZGVDb2x1bW4pO1xuICBjb25zdCBjb2xMZW4gPSBwZXB0aWRlQ29sdW1uLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0RGYgPSBERy5EYXRhRnJhbWUuY3JlYXRlKGNvbExlbik7XG4gIGxldCBtb25vbWVyTGlzdCA9IHNwbGl0dGVyKHBlcHRpZGVDb2x1bW4uZ2V0KDApISk7XG4gIGNvbnN0IGNvbHVtbkxpc3Q6IERHLkNvbHVtbjxzdHJpbmc+W10gPSBbXTtcblxuICAvLyBjcmVhdGUgY29sdW1ucyBhbmQgZmlsbCB0aGUgZmlyc3Qgcm93IGZvciBmYXN0ZXIgdmFsdWVzIGZpbGxpbmcgaW4gdGhlIG5leHQgbG9vcFxuICBmb3IgKGxldCBpID0gMDsgaSA8IG1vbm9tZXJMaXN0Lmxlbmd0aDsgaSsrKSB7XG4gICAgY29uc3QgY29sID0gcmVzdWx0RGYuY29sdW1ucy5hZGROZXdTdHJpbmcoKGkgKyAxKS50b1N0cmluZygpKTtcbiAgICBjb2wuc2V0KDAsIG1vbm9tZXJMaXN0W2ldIHx8ICctJywgZmFsc2UpO1xuICAgIGNvbHVtbkxpc3QucHVzaChjb2wpO1xuICB9XG5cbiAgZm9yIChsZXQgcm93SW5kZXggPSAxOyByb3dJbmRleCA8IGNvbExlbjsgcm93SW5kZXgrKykge1xuICAgIG1vbm9tZXJMaXN0ID0gc3BsaXR0ZXIocGVwdGlkZUNvbHVtbi5nZXQocm93SW5kZXgpISk7XG4gICAgbW9ub21lckxpc3QuZm9yRWFjaCgobW9ub21lciwgY29sSW5kZXgpID0+IGNvbHVtbkxpc3RbY29sSW5kZXhdLnNldChyb3dJbmRleCwgbW9ub21lciB8fCAnLScsIGZhbHNlKSk7XG4gIH1cblxuICByZXR1cm4gcmVzdWx0RGY7XG59XG4gICJdfQ==","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '../viewers/web-logo';\n/** Class for handling notation units in Macromolecule columns */\nexport class UnitsHandler {\n    constructor(col) {\n        this._column = col;\n        const units = this._column.tags[DG.TAGS.UNITS];\n        if (units !== null)\n            this._units = units;\n        else\n            throw new Error('Units are not specified in column');\n        this._notation = this.getNotation();\n        this._defaultGapSymbol = (this.isFasta()) ? UnitsHandler._defaultGapSymbolsDict.FASTA :\n            (this.isHelm()) ? UnitsHandler._defaultGapSymbolsDict.HELM :\n                UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n        if (!this.column.tags.has(UnitsHandler.TAGS.alphabetSize)) {\n            if (this.isHelm())\n                throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n            else if (['UN'].includes(this.alphabet))\n                throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n        }\n        if (!this.column.tags.has(UnitsHandler.TAGS.alphabetIsMultichar)) {\n            if (this.isHelm())\n                throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n            else if (['UN'].includes(this.alphabet))\n                throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n                    `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n        }\n    }\n    static setUnitsToFastaColumn(col) {\n        if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n            throw new Error('Fasta column must be MACROMOLECULE');\n        const stats = WebLogo.getStats(col, 5, WebLogo.splitterAsFasta);\n        const seqType = stats.sameLength ? 'SEQ.MSA' : 'SEQ';\n        const alphabetCandidates = [\n            [\"PT\" /* ALPHABET.PT */, UnitsHandler.PeptideFastaAlphabet],\n            [\"DNA\" /* ALPHABET.DNA */, UnitsHandler.DnaFastaAlphabet],\n            [\"RNA\" /* ALPHABET.RNA */, UnitsHandler.RnaFastaAlphabet],\n        ];\n        // Calculate likelihoods for alphabet_candidates\n        const alphabetCandidatesSim = alphabetCandidates.map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[1]));\n        const maxCos = Math.max(...alphabetCandidatesSim);\n        const alphabet = maxCos > 0.65 ? alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][0] : 'UN';\n        col.setTag(DG.TAGS.UNITS, \"fasta\" /* NOTATION.FASTA */);\n        col.setTag(UnitsHandler.TAGS.aligned, seqType);\n        col.setTag(UnitsHandler.TAGS.alphabet, alphabet);\n    }\n    get units() { return this._units; }\n    get column() { return this._column; }\n    get notation() { return this._notation; }\n    get defaultGapSymbol() { return this._defaultGapSymbol; }\n    get separator() {\n        const separator = this.column.getTag(UnitsHandler.TAGS.separator);\n        if (separator !== null)\n            return separator;\n        else\n            throw new Error('Separator not set');\n    }\n    get aligned() {\n        const aligned = this.column.getTag(UnitsHandler.TAGS.aligned);\n        if (aligned !== null) {\n            return aligned;\n        }\n        else {\n            throw new Error('Tag aligned not set');\n        }\n    }\n    /** Alphabet name (upper case) */\n    get alphabet() {\n        const alphabet = this.column.getTag(UnitsHandler.TAGS.alphabet);\n        if (alphabet !== null) {\n            return alphabet.toUpperCase();\n        }\n        else {\n            throw new Error('Tag alphabet not set');\n        }\n    }\n    getAlphabetSize() {\n        if (this.notation == \"helm\" /* NOTATION.HELM */ || this.alphabet == \"UN\" /* ALPHABET.UN */) {\n            const alphabetSize = parseInt(this.column.getTag(UnitsHandler.TAGS.alphabetSize));\n            return alphabetSize;\n        }\n        else {\n            switch (this.alphabet) {\n                case \"PT\" /* ALPHABET.PT */:\n                    return 20;\n                case \"DNA\" /* ALPHABET.DNA */:\n                case \"RNA\" /* ALPHABET.RNA */:\n                    return 4;\n                case 'NT':\n                    console.warn(`Unexpected alphabet 'NT'.`);\n                    return 4;\n                default:\n                    throw new Error(`Unexpected alphabet '${this.alphabet}'.`);\n            }\n        }\n    }\n    getAlphabetIsMultichar() {\n        if (this.notation == \"helm\" /* NOTATION.HELM */ || this.alphabet == \"UN\" /* ALPHABET.UN */) {\n            return this.column.getTag(UnitsHandler.TAGS.alphabetIsMultichar) == 'true';\n        }\n        else {\n            return false;\n        }\n    }\n    isFasta() { return this.notation === \"fasta\" /* NOTATION.FASTA */; }\n    isSeparator() { return this.notation === \"separator\" /* NOTATION.SEPARATOR */; }\n    isHelm() { return this.notation === \"helm\" /* NOTATION.HELM */; }\n    isRna() { return this.alphabet === 'RNA'; }\n    isDna() { return this.alphabet === 'DNA'; }\n    isPeptide() { return this.alphabet === 'PT'; }\n    /** Associate notation types with the corresponding units */\n    /**\n     * @return {NOTATION}     Notation associated with the units type\n     */\n    getNotation() {\n        if (this.units.toLowerCase().startsWith(\"fasta\" /* NOTATION.FASTA */))\n            return \"fasta\" /* NOTATION.FASTA */;\n        else if (this.units.toLowerCase().startsWith(\"separator\" /* NOTATION.SEPARATOR */))\n            return \"separator\" /* NOTATION.SEPARATOR */;\n        else if (this.units.toLowerCase().startsWith(\"helm\" /* NOTATION.HELM */))\n            return \"helm\" /* NOTATION.HELM */;\n        else\n            throw new Error(`Column '${this.column.name}' has unexpected notation '${this.units}'.`);\n    }\n    /**\n     * Create a new empty column of the specified notation type and the same\n     * length as column\n     *\n     * @param {NOTATION} targetNotation\n     * @return {DG.Column}\n     */\n    getNewColumn(targetNotation) {\n        const col = this.column;\n        const len = col.length;\n        const name = targetNotation.toLowerCase() + '(' + col.name + ')';\n        const newColName = col.dataFrame.columns.getUnusedName(name);\n        const newColumn = DG.Column.fromList('string', newColName, new Array(len).fill(''));\n        newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n        newColumn.setTag(DG.TAGS.UNITS, this.notation);\n        newColumn.setTag(DG.TAGS.CELL_RENDERER, 'Macromolecule');\n        const srcAlphabet = col.getTag(UnitsHandler.TAGS.alphabet);\n        if (srcAlphabet)\n            newColumn.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n        const srcAlphabetSize = col.getTag(UnitsHandler.TAGS.alphabetSize);\n        if (srcAlphabetSize)\n            newColumn.setTag(UnitsHandler.TAGS.alphabetSize, srcAlphabetSize);\n        const srcAlphabetIsMultichar = col.getTag(UnitsHandler.TAGS.alphabetIsMultichar);\n        if (srcAlphabetIsMultichar)\n            newColumn.setTag(UnitsHandler.TAGS.alphabetIsMultichar, srcAlphabetIsMultichar);\n        return newColumn;\n    }\n    /**\n     * Create a new empty column using templateCol as a template\n     *\n     * @param {DG.Column} templateCol  the properties and units of this column are used as a\n     * template to build the new one\n     * @return {DG.Column}\n     */\n    static getNewColumn(templateCol) {\n        const col = new UnitsHandler(templateCol);\n        const targetNotation = col.notation;\n        return col.getNewColumn(targetNotation);\n    }\n    /**\n     * A helper function checking the validity of the 'units' string\n     *\n     * @param {string} units  the string to be validated\n     * @return {boolean}\n     */\n    static unitsStringIsValid(units) {\n        units = units.toLowerCase();\n        const prefixes = [\"fasta\" /* NOTATION.FASTA */, \"separator\" /* NOTATION.SEPARATOR */, \"helm\" /* NOTATION.HELM */];\n        const postfixes = ['rna', 'dna', 'pt'];\n        const prefixCriterion = prefixes.some((p) => units.startsWith(p.toLowerCase()));\n        return prefixCriterion;\n    }\n    /**\n     * Construct a new column of semantic type MACROMOLECULE from the list of\n     * specified parameters\n     *\n     * @param {number}    len  the length of the new column\n     * @param {string}    name  the name of the new column\n     * @param {string}    units  the units of the new column\n     * @return {DG.Column}\n     */\n    static getNewColumnFromParams(len, name, units) {\n        // WARNING: in this implementation is is impossible to verify the uniqueness\n        // of the new column's name\n        // TODO: verify the validity of units parameter\n        if (!UnitsHandler.unitsStringIsValid(units))\n            throw new Error('Invalid format of \\'units\\' parameter');\n        const newColumn = DG.Column.fromList('string', name, new Array(len).fill(''));\n        newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n        newColumn.setTag(DG.TAGS.UNITS, units);\n        return newColumn;\n    }\n}\nUnitsHandler.TAGS = {\n    aligned: 'aligned',\n    alphabet: 'alphabet',\n    alphabetSize: '.alphabetSize',\n    alphabetIsMultichar: '.alphabetIsMultichar',\n    separator: 'separator',\n};\nUnitsHandler._defaultGapSymbolsDict = {\n    HELM: '*',\n    SEPARATOR: '',\n    FASTA: '-',\n};\nUnitsHandler.PeptideFastaAlphabet = new Set([\n    'G', 'L', 'Y', 'S', 'E', 'Q', 'D', 'N', 'F', 'A',\n    'K', 'R', 'H', 'C', 'V', 'P', 'W', 'I', 'M', 'T',\n]);\nUnitsHandler.DnaFastaAlphabet = new Set(['A', 'C', 'G', 'T']);\nUnitsHandler.RnaFastaAlphabet = new Set(['A', 'C', 'G', 'U']);\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"units-handler.js","sourceRoot":"","sources":["units-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport * as grok from 'datagrok-api/grok';\n\nimport {WebLogo, SeqColStats} from '../viewers/web-logo';\n\n/** enum type to simplify setting \"user-friendly\" notation if necessary */\nexport const enum NOTATION {\n  FASTA = 'fasta',\n  SEPARATOR = 'separator',\n  HELM = 'helm',\n}\n\nexport const enum ALPHABET {\n  DNA = 'DNA',\n  RNA = 'RNA',\n  PT = 'PT',\n  UN = 'UN',\n}\n\n/** Class for handling notation units in Macromolecule columns */\nexport class UnitsHandler {\n  public static readonly TAGS = {\n    aligned: 'aligned',\n    alphabet: 'alphabet',\n    alphabetSize: '.alphabetSize',\n    alphabetIsMultichar: '.alphabetIsMultichar',\n    separator: 'separator',\n  };\n\n  protected readonly _column: DG.Column; // the column to be converted\n  protected _units: string; // units, of the form fasta, separator\n  protected _notation: NOTATION; // current notation (without :SEQ:NT, etc.)\n  protected _defaultGapSymbol: string;\n  protected static readonly _defaultGapSymbolsDict = {\n    HELM: '*',\n    SEPARATOR: '',\n    FASTA: '-',\n  };\n\n  public static readonly PeptideFastaAlphabet = new Set([\n    'G', 'L', 'Y', 'S', 'E', 'Q', 'D', 'N', 'F', 'A',\n    'K', 'R', 'H', 'C', 'V', 'P', 'W', 'I', 'M', 'T',\n  ]);\n  public static readonly DnaFastaAlphabet = new Set(['A', 'C', 'G', 'T']);\n  public static readonly RnaFastaAlphabet = new Set(['A', 'C', 'G', 'U']);\n\n  public static setUnitsToFastaColumn(col: DG.Column) {\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n      throw new Error('Fasta column must be MACROMOLECULE');\n\n    const stats: SeqColStats = WebLogo.getStats(col, 5, WebLogo.splitterAsFasta);\n    const seqType = stats.sameLength ? 'SEQ.MSA' : 'SEQ';\n\n    const alphabetCandidates: [string, Set<string>][] = [\n      [ALPHABET.PT, UnitsHandler.PeptideFastaAlphabet],\n      [ALPHABET.DNA, UnitsHandler.DnaFastaAlphabet],\n      [ALPHABET.RNA, UnitsHandler.RnaFastaAlphabet],\n    ];\n\n    // Calculate likelihoods for alphabet_candidates\n    const alphabetCandidatesSim: number[] = alphabetCandidates.map(\n      (c) => WebLogo.getAlphabetSimilarity(stats.freq, c[1]));\n    const maxCos = Math.max(...alphabetCandidatesSim);\n    const alphabet = maxCos > 0.65 ? alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][0] : 'UN';\n\n    col.setTag(DG.TAGS.UNITS, NOTATION.FASTA);\n    col.setTag(UnitsHandler.TAGS.aligned, seqType);\n    col.setTag(UnitsHandler.TAGS.alphabet, alphabet);\n  }\n\n  protected get units(): string { return this._units; }\n\n  protected get column(): DG.Column { return this._column; }\n\n  public get notation(): NOTATION { return this._notation; }\n\n  public get defaultGapSymbol(): string { return this._defaultGapSymbol; }\n\n  public get separator(): string {\n    const separator = this.column.getTag(UnitsHandler.TAGS.separator);\n    if (separator !== null)\n      return separator;\n    else\n      throw new Error('Separator not set');\n  }\n\n  public get aligned(): string {\n    const aligned = this.column.getTag(UnitsHandler.TAGS.aligned);\n    if (aligned !== null) {\n      return aligned;\n    } else {\n      throw new Error('Tag aligned not set');\n    }\n  }\n\n  /** Alphabet name (upper case) */\n  public get alphabet(): string {\n    const alphabet = this.column.getTag(UnitsHandler.TAGS.alphabet);\n    if (alphabet !== null) {\n      return alphabet.toUpperCase();\n    } else {\n      throw new Error('Tag alphabet not set');\n    }\n  }\n\n  public getAlphabetSize(): number {\n    if (this.notation == NOTATION.HELM || this.alphabet == ALPHABET.UN) {\n      const alphabetSize = parseInt(this.column.getTag(UnitsHandler.TAGS.alphabetSize));\n      return alphabetSize;\n    } else {\n      switch (this.alphabet) {\n      case ALPHABET.PT:\n        return 20;\n      case ALPHABET.DNA:\n      case ALPHABET.RNA:\n        return 4;\n      case 'NT':\n        console.warn(`Unexpected alphabet 'NT'.`);\n        return 4;\n      default:\n        throw new Error(`Unexpected alphabet '${this.alphabet}'.`);\n      }\n    }\n  }\n\n  public getAlphabetIsMultichar(): boolean {\n    if (this.notation == NOTATION.HELM || this.alphabet == ALPHABET.UN) {\n      return this.column.getTag(UnitsHandler.TAGS.alphabetIsMultichar) == 'true';\n    } else {\n      return false;\n    }\n  }\n\n  public isFasta(): boolean { return this.notation === NOTATION.FASTA; }\n\n  public isSeparator(): boolean { return this.notation === NOTATION.SEPARATOR; }\n\n  public isHelm(): boolean { return this.notation === NOTATION.HELM; }\n\n  public isRna(): boolean { return this.alphabet === 'RNA'; }\n\n  public isDna(): boolean { return this.alphabet === 'DNA'; }\n\n  public isPeptide(): boolean { return this.alphabet === 'PT'; }\n\n  /** Associate notation types with the corresponding units */\n  /**\n   * @return {NOTATION}     Notation associated with the units type\n   */\n  protected getNotation(): NOTATION {\n    if (this.units.toLowerCase().startsWith(NOTATION.FASTA))\n      return NOTATION.FASTA;\n    else if (this.units.toLowerCase().startsWith(NOTATION.SEPARATOR))\n      return NOTATION.SEPARATOR;\n    else if (this.units.toLowerCase().startsWith(NOTATION.HELM))\n      return NOTATION.HELM;\n    else\n      throw new Error(`Column '${this.column.name}' has unexpected notation '${this.units}'.`);\n  }\n\n  /**\n   * Create a new empty column of the specified notation type and the same\n   * length as column\n   *\n   * @param {NOTATION} targetNotation\n   * @return {DG.Column}\n   */\n  protected getNewColumn(targetNotation: NOTATION): DG.Column {\n    const col = this.column;\n    const len = col.length;\n    const name = targetNotation.toLowerCase() + '(' + col.name + ')';\n    const newColName = col.dataFrame.columns.getUnusedName(name);\n    const newColumn = DG.Column.fromList('string', newColName, new Array(len).fill(''));\n    newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n    newColumn.setTag(DG.TAGS.UNITS, this.notation);\n    newColumn.setTag(DG.TAGS.CELL_RENDERER, 'Macromolecule');\n\n    const srcAlphabet = col.getTag(UnitsHandler.TAGS.alphabet);\n    if (srcAlphabet)\n      newColumn.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n\n    const srcAlphabetSize = col.getTag(UnitsHandler.TAGS.alphabetSize);\n    if (srcAlphabetSize)\n      newColumn.setTag(UnitsHandler.TAGS.alphabetSize, srcAlphabetSize);\n\n    const srcAlphabetIsMultichar = col.getTag(UnitsHandler.TAGS.alphabetIsMultichar);\n    if (srcAlphabetIsMultichar)\n      newColumn.setTag(UnitsHandler.TAGS.alphabetIsMultichar, srcAlphabetIsMultichar);\n\n    return newColumn;\n  }\n\n  /**\n   * Create a new empty column using templateCol as a template\n   *\n   * @param {DG.Column} templateCol  the properties and units of this column are used as a\n   * template to build the new one\n   * @return {DG.Column}\n   */\n  public static getNewColumn(templateCol: DG.Column): DG.Column {\n    const col: UnitsHandler = new UnitsHandler(templateCol);\n    const targetNotation = col.notation;\n    return col.getNewColumn(targetNotation);\n  }\n\n  /**\n   * A helper function checking the validity of the 'units' string\n   *\n   * @param {string} units  the string to be validated\n   * @return {boolean}\n   */\n  public static unitsStringIsValid(units: string): boolean {\n    units = units.toLowerCase();\n    const prefixes = [NOTATION.FASTA, NOTATION.SEPARATOR, NOTATION.HELM];\n    const postfixes = ['rna', 'dna', 'pt'];\n\n    const prefixCriterion = prefixes.some((p) => units.startsWith(p.toLowerCase()));\n    return prefixCriterion;\n  }\n\n  /**\n   * Construct a new column of semantic type MACROMOLECULE from the list of\n   * specified parameters\n   *\n   * @param {number}    len  the length of the new column\n   * @param {string}    name  the name of the new column\n   * @param {string}    units  the units of the new column\n   * @return {DG.Column}\n   */\n  public static getNewColumnFromParams(\n    len: number,\n    name: string,\n    units: string\n  ): DG.Column {\n    // WARNING: in this implementation is is impossible to verify the uniqueness\n    // of the new column's name\n    // TODO: verify the validity of units parameter\n    if (!UnitsHandler.unitsStringIsValid(units))\n      throw new Error('Invalid format of \\'units\\' parameter');\n    const newColumn = DG.Column.fromList('string', name, new Array(len).fill(''));\n    newColumn.semType = DG.SEMTYPE.MACROMOLECULE;\n    newColumn.setTag(DG.TAGS.UNITS, units);\n    return newColumn;\n  }\n\n  public constructor(col: DG.Column) {\n    this._column = col;\n    const units = this._column.tags[DG.TAGS.UNITS];\n    if (units !== null)\n      this._units = units;\n    else\n      throw new Error('Units are not specified in column');\n    this._notation = this.getNotation();\n    this._defaultGapSymbol = (this.isFasta()) ? UnitsHandler._defaultGapSymbolsDict.FASTA :\n      (this.isHelm()) ? UnitsHandler._defaultGapSymbolsDict.HELM :\n        UnitsHandler._defaultGapSymbolsDict.SEPARATOR;\n\n    if (!this.column.tags.has(UnitsHandler.TAGS.alphabetSize)) {\n      if (this.isHelm())\n        throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n      else if (['UN'].includes(this.alphabet))\n        throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetSize}' is mandatory.`);\n    }\n\n    if (!this.column.tags.has(UnitsHandler.TAGS.alphabetIsMultichar)) {\n      if (this.isHelm())\n        throw new Error(`For column '${this.column.name}' of notation '${this.notation}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n      else if (['UN'].includes(this.alphabet))\n        throw new Error(`For column '${this.column.name}' of alphabet '${this.alphabet}' ` +\n          `tag '${UnitsHandler.TAGS.alphabetIsMultichar}' is mandatory.`);\n    }\n  }\n}\n"]}","// Data structures for V-Domain regions of antibodies\nexport var VdRegionType;\n(function (VdRegionType) {\n    VdRegionType[\"Unknown\"] = \"unknown\";\n    VdRegionType[\"FR\"] = \"framework\";\n    VdRegionType[\"CDR\"] = \"cdr\";\n})(VdRegionType || (VdRegionType = {}));\n/** Describes V-DOMAIN (IG and TR) region (of multiple alignment)\n * https://www.imgt.org/IMGTScientificChart/Numbering/IMGTIGVLsuperfamily.html\n */\nexport class VdRegion {\n    /**\n     * start and position are strings because they correspond to position names as column names in ANARCI output\n     * @param {VdRegionType} type  Type of the region\n     * @param {string} name  Name of the region\n     * @param {string} chain\n     * @param {string} order\n     * @param {string} positionStartName  Region start position (inclusive)\n     * @param {string} positionEndName  Region end position (inclusive)\n     */\n    constructor(type, name, chain, order, positionStartName, positionEndName) {\n        this.type = type;\n        this.name = name;\n        this.chain = chain;\n        this.order = order;\n        this.positionStartName = positionStartName;\n        this.positionEndName = positionEndName;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmQtcmVnaW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInZkLXJlZ2lvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEscURBQXFEO0FBRXJELE1BQU0sQ0FBTixJQUFZLFlBSVg7QUFKRCxXQUFZLFlBQVk7SUFDdEIsbUNBQW1CLENBQUE7SUFDbkIsZ0NBQWdCLENBQUE7SUFDaEIsMkJBQVcsQ0FBQTtBQUNiLENBQUMsRUFKVyxZQUFZLEtBQVosWUFBWSxRQUl2QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFFBQVE7SUFRbkI7Ozs7Ozs7O09BUUc7SUFDSCxZQUFZLElBQWtCLEVBQUUsSUFBWSxFQUFFLEtBQWEsRUFBRSxLQUFhLEVBQ3hFLGlCQUF5QixFQUFFLGVBQXVCO1FBQ2xELElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxpQkFBaUIsR0FBRyxpQkFBaUIsQ0FBQztRQUMzQyxJQUFJLENBQUMsZUFBZSxHQUFHLGVBQWUsQ0FBQztJQUN6QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBEYXRhIHN0cnVjdHVyZXMgZm9yIFYtRG9tYWluIHJlZ2lvbnMgb2YgYW50aWJvZGllc1xuXG5leHBvcnQgZW51bSBWZFJlZ2lvblR5cGUge1xuICBVbmtub3duID0gJ3Vua25vd24nLFxuICBGUiA9ICdmcmFtZXdvcmsnLFxuICBDRFIgPSAnY2RyJyxcbn1cblxuLyoqIERlc2NyaWJlcyBWLURPTUFJTiAoSUcgYW5kIFRSKSByZWdpb24gKG9mIG11bHRpcGxlIGFsaWdubWVudClcbiAqIGh0dHBzOi8vd3d3LmltZ3Qub3JnL0lNR1RTY2llbnRpZmljQ2hhcnQvTnVtYmVyaW5nL0lNR1RJR1ZMc3VwZXJmYW1pbHkuaHRtbFxuICovXG5leHBvcnQgY2xhc3MgVmRSZWdpb24ge1xuICB0eXBlOiBWZFJlZ2lvblR5cGU7XG4gIG5hbWU6IHN0cmluZztcbiAgY2hhaW46IHN0cmluZztcbiAgb3JkZXI6IG51bWJlcjtcbiAgcG9zaXRpb25TdGFydE5hbWU6IHN0cmluZztcbiAgcG9zaXRpb25FbmROYW1lOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIHN0YXJ0IGFuZCBwb3NpdGlvbiBhcmUgc3RyaW5ncyBiZWNhdXNlIHRoZXkgY29ycmVzcG9uZCB0byBwb3NpdGlvbiBuYW1lcyBhcyBjb2x1bW4gbmFtZXMgaW4gQU5BUkNJIG91dHB1dFxuICAgKiBAcGFyYW0ge1ZkUmVnaW9uVHlwZX0gdHlwZSAgVHlwZSBvZiB0aGUgcmVnaW9uXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBuYW1lICBOYW1lIG9mIHRoZSByZWdpb25cbiAgICogQHBhcmFtIHtzdHJpbmd9IGNoYWluXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBvcmRlclxuICAgKiBAcGFyYW0ge3N0cmluZ30gcG9zaXRpb25TdGFydE5hbWUgIFJlZ2lvbiBzdGFydCBwb3NpdGlvbiAoaW5jbHVzaXZlKVxuICAgKiBAcGFyYW0ge3N0cmluZ30gcG9zaXRpb25FbmROYW1lICBSZWdpb24gZW5kIHBvc2l0aW9uIChpbmNsdXNpdmUpXG4gICAqL1xuICBjb25zdHJ1Y3Rvcih0eXBlOiBWZFJlZ2lvblR5cGUsIG5hbWU6IHN0cmluZywgY2hhaW46IHN0cmluZywgb3JkZXI6IG51bWJlcixcbiAgICBwb3NpdGlvblN0YXJ0TmFtZTogc3RyaW5nLCBwb3NpdGlvbkVuZE5hbWU6IHN0cmluZykge1xuICAgIHRoaXMudHlwZSA9IHR5cGU7XG4gICAgdGhpcy5uYW1lID0gbmFtZTtcbiAgICB0aGlzLmNoYWluID0gY2hhaW47XG4gICAgdGhpcy5vcmRlciA9IG9yZGVyO1xuICAgIHRoaXMucG9zaXRpb25TdGFydE5hbWUgPSBwb3NpdGlvblN0YXJ0TmFtZTtcbiAgICB0aGlzLnBvc2l0aW9uRW5kTmFtZSA9IHBvc2l0aW9uRW5kTmFtZTtcbiAgfVxufVxuIl19","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport wu from 'wu';\nimport * as rxjs from 'rxjs';\nimport { Vector } from '@datagrok-libraries/utils/src/type-declarations';\nimport { vectorLength, vectorDotProduct } from '@datagrok-libraries/utils/src/vector-operations';\nimport { Aminoacids, AminoacidsPalettes } from '../aminoacids';\nimport { Nucleotides, NucleotidesPalettes } from '../nucleotides';\nimport { UnknownSeqPalettes } from '../unknown';\nimport { UnitsHandler } from '../utils/units-handler';\nvar PositionHeight;\n(function (PositionHeight) {\n    PositionHeight[\"Entropy\"] = \"Entropy\";\n    PositionHeight[\"full\"] = \"100%\";\n})(PositionHeight || (PositionHeight = {}));\n/**@param {MouseEvent} event\n * @param {number} r devicePixelRation\n * @return {DG.Point} canvas related cursor position\n */\nHTMLCanvasElement.prototype.getCursorPosition = function (event, r) {\n    const rect = this.getBoundingClientRect();\n    return new DG.Point((event.clientX - rect.left) * r, (event.clientY - rect.top) * r);\n};\nDG.Rect.prototype.contains = function (x, y) {\n    return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;\n};\nexport class PositionMonomerInfo {\n    constructor(count = 0, bounds = new DG.Rect(0, 0, 0, 0)) {\n        this.count = count;\n        this.bounds = bounds;\n    }\n}\nexport class PositionInfo {\n    /** freq = {}, rowCount = 0\n     * @param {string} name Name of position ('111A', '111.1', etc)\n     * @param {number} sumForHeightCalc Sum of all monomer counts for height calculation\n     * @param {number} rowCount Count of elements in column\n     * @param {string[]} freq frequency of monomers in position\n     */\n    constructor(name, freq = {}, rowCount = 0, sumForHeightCalc = 0) {\n        this.name = name;\n        this.freq = freq;\n        this.rowCount = rowCount;\n        this.sumForHeightCalc = sumForHeightCalc;\n    }\n}\nexport class WebLogo extends DG.JsViewer {\n    constructor() {\n        super();\n        this.viewerId = -1;\n        this.initialized = false;\n        // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];\n        this.cp = null;\n        this.axisHeight = 12;\n        this.seqCol = null;\n        this.splitter = null;\n        // private maxLength: number = 100;\n        this.positions = [];\n        this.rowsMasked = 0;\n        this.rowsNull = 0;\n        this.positionMargin = 0;\n        this.positionNames = [];\n        this.startPosition = -1;\n        this.endPosition = -1;\n        this.viewSubs = [];\n        this.viewerId = WebLogo.viewerCount;\n        WebLogo.viewerCount += 1;\n        this.textBaseline = 'top';\n        this.unitsHandler = null;\n        this._positionWidth = this.positionWidth = this.float('positionWidth', 16 /*,\n          {editor: 'slider', min: 4, max: 64, postfix: 'px'}*/);\n        this.minHeight = this.float('minHeight', 50 /*,\n          {editor: 'slider', min: 25, max: 250, postfix: 'px'}*/);\n        this.maxHeight = this.float('maxHeight', 100 /*,\n          {editor: 'slider', min: 25, max: 500, postfix: 'px'}*/);\n        this.considerNullSequences = this.bool('considerNullSequences', false);\n        this.sequenceColumnName = this.string('sequenceColumnName', null);\n        this.startPositionName = this.string('startPositionName', null);\n        this.endPositionName = this.string('endPositionName', null);\n        this.fixWidth = this.bool('fixWidth', false);\n        this.verticalAlignment = this.string('verticalAlignment', 'middle', { choices: ['top', 'middle', 'bottom'] });\n        this.horizontalAlignment = this.string('horizontalAlignment', 'center', { choices: ['left', 'center', 'right'] });\n        this.fitArea = this.bool('fitArea', true);\n        this.shrinkEmptyTail = this.bool('shrinkEmptyTail', true);\n        this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n        this.positionMarginState = this.string('positionMarginState', 'auto', { choices: ['auto', 'enable', 'off'] });\n        let defaultValueForPositionMargin = 0;\n        if (this.positionMarginState === 'auto') {\n            defaultValueForPositionMargin = 4;\n        }\n        this.positionMargin = this.int('positionMargin', defaultValueForPositionMargin, { min: 0, max: 16 });\n        this.positionHeight = this.string('positionHeight', PositionHeight.full, { choices: [PositionHeight.full, PositionHeight.Entropy] });\n    }\n    /** For startPosition equals to endPosition Length is 1 */\n    get Length() {\n        if (this.skipEmptyPositions) {\n            return this.positions.length;\n        }\n        return this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0;\n    }\n    get positionWidthWithMargin() {\n        var _a;\n        if ((this.positionMarginState === 'auto') && (((_a = this.unitsHandler) === null || _a === void 0 ? void 0 : _a.getAlphabetIsMultichar()) === true)) {\n            return this._positionWidth + this.positionMargin;\n        }\n        if (this.positionMarginState === 'enable') {\n            return this._positionWidth + this.positionMargin;\n        }\n        return this._positionWidth;\n    }\n    init() {\n        return __awaiter(this, void 0, void 0, function* () {\n            if (this.initialized) {\n                console.error('WebLogo second initialization!');\n                return;\n            }\n            this.initialized = true;\n            this.helpUrl = '/help/visualize/viewers/web-logo.md';\n            this.msgHost = ui.div('No message');\n            this.msgHost.style.display = 'none';\n            this.canvas = ui.canvas();\n            this.canvas.style.width = '100%';\n            this.host = ui.div([this.msgHost, this.canvas]);\n            // this.slider = ui.rangeSlider(0, 20, 2, 5);\n            // this.slider.root.style.width = '100%';\n            // this.slider.root.style.height = '12px';\n            const getMonomer = (p) => {\n                const jPos = Math.floor(p.x / this.positionWidthWithMargin);\n                const position = this.positions[jPos];\n                if (position === void 0)\n                    return [jPos, null, null];\n                const monomer = Object.keys(position.freq)\n                    .find((m) => position.freq[m].bounds.contains(p.x, p.y));\n                if (monomer === undefined)\n                    return [jPos, null, null];\n                return [jPos, monomer, position.freq[monomer]];\n            };\n            // this.subs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mouseover').subscribe((e: MouseEvent) => {\n            // }));\n            this.viewSubs.push(rxjs.fromEvent(this.canvas, 'mousemove').subscribe((e) => {\n                if (!this.canvas)\n                    return;\n                const args = e;\n                const r = window.devicePixelRatio;\n                const cursorP = this.canvas.getCursorPosition(args, r);\n                const [jPos, monomer] = getMonomer(cursorP);\n                if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n                    const rowCount = wu.count().take(this.dataFrame.rowCount).filter((iRow) => {\n                        const seq = this.seqCol.get(iRow);\n                        const seqM = seq ? this.splitter(seq)[this.startPosition + jPos] : null;\n                        return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n                    }).reduce((count, iRow) => count + 1, 0);\n                    ui.tooltip.show(ui.div([ui.div(`${monomer}`), ui.div(`${rowCount} rows`)]), args.x + 16, args.y + 16);\n                }\n                else {\n                    ui.tooltip.hide();\n                }\n            }));\n            this.viewSubs.push(rxjs.fromEvent(this.canvas, 'mousedown').subscribe((e) => {\n                if (!this.canvas || e.button != 0)\n                    return;\n                const args = e;\n                const r = window.devicePixelRatio;\n                const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args, r));\n                // prevents deselect all rows if we miss monomer bounds\n                if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n                    this.dataFrame.selection.init((iRow) => {\n                        const seq = this.seqCol.get(iRow);\n                        const seqM = seq ? this.splitter(seq)[this.startPosition + jPos] : null;\n                        return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n                    });\n                }\n            }));\n            this.viewSubs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n            this.root.append(this.host);\n            // this.root.appendChild(this.slider.root);\n            this.render(true);\n        });\n    }\n    rootOnSizeChanged(args) {\n        this.render(true);\n        // console.debug(`WebLogo.onRootSizeChanged() ` +\n        //   `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +\n        //   `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);\n    }\n    /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().\n     */\n    updateSeqCol() {\n        if (this.dataFrame) {\n            this.seqCol = this.sequenceColumnName ? this.dataFrame.col(this.sequenceColumnName) : null;\n            if (this.seqCol == null) {\n                this.seqCol = WebLogo.pickUpSeqCol2(this.dataFrame);\n                this.sequenceColumnName = this.seqCol ? this.seqCol.name : null;\n            }\n            if (this.seqCol) {\n                const units = this.seqCol.getTag(DG.TAGS.UNITS);\n                const separator = this.seqCol.getTag(UnitsHandler.TAGS.separator);\n                this.splitter = WebLogo.getSplitter(units, separator);\n                this.unitsHandler = new UnitsHandler(this.seqCol);\n                this.updatePositions();\n                this.cp = WebLogo.pickUpPalette(this.seqCol);\n            }\n            else {\n                this.splitter = null;\n                this.positionNames = [];\n                this.startPosition = -1;\n                this.endPosition = -1;\n                this.cp = null;\n            }\n        }\n        this.render();\n    }\n    updatePositions() {\n        if (!this.seqCol)\n            return;\n        let categories;\n        if (this.shrinkEmptyTail) {\n            const indices = this.dataFrame.filter.getSelectedIndexes();\n            categories = Array.from(new Set(Array.from(Array(indices.length).keys()).map((i) => this.seqCol.get(indices[i]))));\n        }\n        else {\n            categories = this.seqCol.categories;\n        }\n        const maxLength = categories.length > 0 ? Math.max(...categories.map((s) => s !== null ? this.splitter(s).length : 0)) : 0;\n        // Get position names from data column tag 'positionNames'\n        const positionNamesTxt = this.seqCol.getTag('positionNames');\n        // Fallback if 'positionNames' tag is not provided\n        this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :\n            [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);\n        this.startPosition = (this.startPositionName && this.positionNames &&\n            this.positionNames.includes(this.startPositionName)) ?\n            this.positionNames.indexOf(this.startPositionName) : 0;\n        this.endPosition = (this.endPositionName && this.positionNames &&\n            this.positionNames.includes(this.endPositionName)) ?\n            this.positionNames.indexOf(this.endPositionName) : (maxLength - 1);\n    }\n    onPropertyChanged(property) {\n        // console.debug(`WebLogo.onPropertyChanged( ${property.name} = '' })`);\n        super.onPropertyChanged(property);\n        switch (property.name) {\n            case 'considerNullSequences':\n                this.render(true);\n                break;\n            case 'sequenceColumnName':\n                this.updateSeqCol();\n                break;\n            case 'startPositionName':\n                this.updateSeqCol();\n                break;\n            case 'endPositionName':\n                this.updateSeqCol();\n                break;\n            case 'positionWidth':\n                this._positionWidth = this.positionWidth;\n                this.render(true);\n                break;\n            case 'minHeight':\n                this.render(true);\n                break;\n            case 'maxHeight':\n                this.render(true);\n                break;\n            case 'fixWidth':\n                this.render(true);\n                break;\n            case 'verticalAlignment':\n                this.render(true);\n                break;\n            case 'horizontalAlignment':\n                this.render(true);\n                break;\n            case 'fitArea':\n                this.render(true);\n                break;\n            case 'shrinkEmptyTail':\n                this.updatePositions();\n                this.render(true);\n                break;\n            case 'skipEmptyPositions':\n                this.updatePositions();\n                this.render(true);\n                break;\n            case 'positionMargin':\n                this.render(true);\n                break;\n            case 'positionMarginState':\n                this.render(true);\n                break;\n            case 'positionHeight':\n                this.render(true);\n                break;\n        }\n    }\n    onTableAttached() {\n        const _super = Object.create(null, {\n            onTableAttached: { get: () => super.onTableAttached }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            const dataFrameTxt = this.dataFrame ? 'data' : 'null';\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n            _super.onTableAttached.call(this);\n            this.updateSeqCol();\n            if (this.dataFrame !== void 0) {\n                this.subs.push(this.dataFrame.selection.onChanged.subscribe((_) => this.render()));\n                this.subs.push(this.dataFrame.filter.onChanged.subscribe((_) => {\n                    this.updatePositions();\n                    this.render();\n                }));\n            }\n            yield this.init();\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n        });\n    }\n    detach() {\n        const _super = Object.create(null, {\n            detach: { get: () => super.detach }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            const dataFrameTxt = `${this.dataFrame ? 'data' : 'null'}`;\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n            _super.detach.call(this);\n            this.viewSubs.forEach((sub) => sub.unsubscribe());\n            this.host.remove();\n            this.canvas = undefined;\n            this.msgHost = undefined;\n            this.host = undefined;\n            this.initialized = false;\n            console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n        });\n    }\n    _nullSequence(fillerResidue = 'X') {\n        if (this.considerNullSequences)\n            return new Array(this.Length).fill(fillerResidue).join('');\n        return '';\n    }\n    removeWhere(array, predicate) {\n        let length = array.length;\n        let updateIterator = 0;\n        for (let deleteIterator = 0; deleteIterator < length; deleteIterator++) {\n            if (!predicate(array[deleteIterator])) {\n                array[updateIterator] = array[deleteIterator];\n                updateIterator++;\n            }\n        }\n        array.length = updateIterator;\n        return array;\n    }\n    _removeEmptyPositions() {\n        if (this.skipEmptyPositions) {\n            this.removeWhere(this.positions, item => { var _a; return ((_a = item === null || item === void 0 ? void 0 : item.freq['-']) === null || _a === void 0 ? void 0 : _a.count) === item.rowCount; });\n        }\n    }\n    _calculate(r) {\n        if (!this.canvas || !this.host || !this.seqCol || !this.dataFrame)\n            return;\n        this.unitsHandler = new UnitsHandler(this.seqCol);\n        this.calcSize();\n        this.positions = new Array(this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0);\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const posName = this.positionNames[this.startPosition + jPos];\n            this.positions[jPos] = new PositionInfo(posName);\n        }\n        // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)\n        const indices = this.dataFrame.filter.getSelectedIndexes();\n        // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :\n        //   this.dataFrame.filter.getSelectedIndexes();\n        this.rowsMasked = indices.length;\n        this.rowsNull = 0;\n        for (const i of indices) {\n            let s = (this.seqCol.get(i));\n            if (!s) {\n                s = this._nullSequence();\n                ++this.rowsNull;\n            }\n            const seqM = this.splitter(s);\n            for (let jPos = 0; jPos < this.Length; jPos++) {\n                const pmInfo = this.positions[jPos].freq;\n                const m = seqM[this.startPosition + jPos] || '-';\n                if (!(m in pmInfo))\n                    pmInfo[m] = new PositionMonomerInfo();\n                pmInfo[m].count++;\n            }\n        }\n        //#region Polish freq counts\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            // delete this.positions[jPos].freq['-'];\n            this.positions[jPos].rowCount = 0;\n            for (const m in this.positions[jPos].freq)\n                this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;\n            if (this.positionHeight == PositionHeight.Entropy) {\n                this.positions[jPos].sumForHeightCalc = 0;\n                for (const m in this.positions[jPos].freq) {\n                    const pn = this.positions[jPos].freq[m].count / this.positions[jPos].rowCount;\n                    this.positions[jPos].sumForHeightCalc += -pn * Math.log2(pn);\n                }\n            }\n        }\n        //#endregion\n        this._removeEmptyPositions();\n        const absoluteMaxHeight = this.canvas.height - this.axisHeight * r;\n        // console.debug(`WebLogo<${this.viewerId}>._calculate() maxHeight=${maxHeight}.`);\n        //#region Calculate screen\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const freq = this.positions[jPos].freq;\n            const rowCount = this.positions[jPos].rowCount;\n            const alphabetSize = this.getAlphabetSize();\n            if ((this.positionHeight == PositionHeight.Entropy) && (alphabetSize == null)) {\n                grok.shell.error('WebLogo: alphabet is undefined.');\n            }\n            const maxHeight = (this.positionHeight == PositionHeight.Entropy) ? (absoluteMaxHeight * (Math.log2(alphabetSize) - (this.positions[jPos].sumForHeightCalc)) / Math.log2(alphabetSize)) : absoluteMaxHeight;\n            let y = this.axisHeight * r + (absoluteMaxHeight - maxHeight - 1);\n            const entries = Object.entries(freq).sort((a, b) => {\n                if (a[0] !== '-' && b[0] !== '-')\n                    return b[1].count - a[1].count;\n                else if (a[0] === '-' && b[0] === '-')\n                    return 0;\n                else if (a[0] === '-')\n                    return -1;\n                else /* (b[0] === '-') */\n                    return +1;\n            });\n            for (const entry of entries) {\n                const pmInfo = entry[1];\n                // const m: string = entry[0];\n                const h = maxHeight * pmInfo.count / rowCount;\n                pmInfo.bounds = new DG.Rect(jPos * this.positionWidthWithMargin, y, this._positionWidth, h);\n                y += h;\n            }\n        }\n        //#endregion\n    }\n    // reflect changes made to filter/selection\n    render(recalc = true) {\n        var _a;\n        if (this.msgHost) {\n            if (this.seqCol && !this.cp) {\n                this.msgHost.innerText = `Unknown palette (column semType: '${this.seqCol.semType}').`;\n                this.msgHost.style.display = '';\n            }\n            else {\n                this.msgHost.style.display = 'none';\n            }\n        }\n        if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp)\n            return;\n        const g = this.canvas.getContext('2d');\n        if (!g)\n            return;\n        const r = window.devicePixelRatio;\n        if (recalc)\n            this._calculate(r);\n        g.resetTransform();\n        g.fillStyle = 'white';\n        g.fillRect(0, 0, this.canvas.width, this.canvas.height);\n        g.textBaseline = this.textBaseline;\n        // Prevents division by zero on Length = 0\n        if (this.startPosition === -1 || this.endPosition === -1)\n            return;\n        //#region Plot positionNames\n        const positionFontSize = 10 * r;\n        g.resetTransform();\n        g.fillStyle = 'black';\n        g.textAlign = 'center';\n        g.font = `${positionFontSize.toFixed(1)}px Roboto, Roboto Local, sans-serif`;\n        const posNameMaxWidth = Math.max(...this.positions.map((pos) => g.measureText(pos.name).width));\n        const hScale = posNameMaxWidth < (this._positionWidth - 2) ? 1 : (this._positionWidth - 2) / posNameMaxWidth;\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            const pos = this.positions[jPos];\n            g.resetTransform();\n            g.setTransform(hScale, 0, 0, 1, jPos * this.positionWidthWithMargin + this._positionWidth / 2, 0);\n            g.fillText(pos.name, 0, 0);\n        }\n        //#endregion Plot positionNames\n        for (let jPos = 0; jPos < this.Length; jPos++) {\n            for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {\n                if (monomer !== '-') {\n                    const monomerTxt = WebLogo.monomerToText(monomer);\n                    const b = pmInfo.bounds;\n                    const fontStyle = '16px Roboto, Roboto Local, sans-serif';\n                    // Hacks to scale uppercase characters to target rectangle\n                    const uppercaseLetterAscent = 0.25;\n                    const uppercaseLetterHeight = 12.2;\n                    g.resetTransform();\n                    g.strokeStyle = 'lightgray';\n                    g.lineWidth = 1;\n                    g.rect(b.left, b.top, b.width, b.height);\n                    g.fillStyle = (_a = this.cp.get(monomer)) !== null && _a !== void 0 ? _a : this.cp.get('other');\n                    g.textAlign = 'left';\n                    g.font = fontStyle;\n                    //g.fillRect(b.left, b.top, b.width, b.height);\n                    const mTm = g.measureText(monomerTxt);\n                    // if (mM.actualBoundingBoxAscent != 0)\n                    //   console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);\n                    g.setTransform(b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight, b.left, b.top);\n                    g.fillText(monomerTxt, 0, -uppercaseLetterAscent);\n                }\n            }\n        }\n    }\n    calcSize() {\n        if (!this.canvas || !this.host)\n            return;\n        const r = window.devicePixelRatio;\n        let width = this.Length * this.positionWidthWithMargin / r;\n        let height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));\n        if (this.fitArea) {\n            const xScale = this.root.clientHeight / height;\n            const yScale = this.root.clientWidth / width;\n            const scale = Math.max(1, Math.min(xScale, yScale));\n            width = width * scale;\n            height = height * scale;\n            this._positionWidth = this.positionWidth * scale;\n        }\n        this.canvas.width = width * r;\n        this.canvas.style.width = `${width}px`;\n        // const canvasHeight: number = width > this.root.clientWidth ? height - 8 : height;\n        this.host.style.setProperty('height', `${height}px`);\n        const canvasHeight = this.host.clientHeight;\n        this.canvas.height = canvasHeight * r;\n        this.canvas.style.setProperty('height', `${canvasHeight}px`);\n        // Adjust host and root width\n        if (this.fixWidth) {\n            // full width for canvas host and root\n            this.root.style.width = this.host.style.width = `${width}px`;\n            this.root.style.height = `${height}px`;\n            this.host.style.setProperty('overflow', 'hidden', 'important');\n        }\n        else {\n            // allow scroll canvas in root\n            this.root.style.width = this.host.style.width = '100%';\n            this.host.style.overflowX = 'auto!important';\n            this.host.style.setProperty('overflow', null);\n            this.host.style.setProperty('text-align', this.horizontalAlignment);\n            // vertical alignment\n            let hostTopMargin = 0;\n            switch (this.verticalAlignment) {\n                case 'top':\n                    hostTopMargin = 0;\n                    break;\n                case 'middle':\n                    hostTopMargin = Math.max(0, (this.root.clientHeight - height) / 2);\n                    break;\n                case 'bottom':\n                    hostTopMargin = Math.max(0, this.root.clientHeight - height);\n                    break;\n            }\n            this.host.style.setProperty('margin-top', `${hostTopMargin}px`, 'important');\n            if (this.root.clientHeight < height) {\n                this.host.style.setProperty('height', `${this.root.clientHeight}px`);\n                this.host.style.setProperty('overflow-y', null);\n            }\n            else {\n                this.host.style.setProperty('overflow-y', 'hidden', 'important');\n            }\n        }\n        // console.debug(\n        //   `this.root.style.height = ${this.root.style.height}\\n` +\n        //   `this.root.clientHeight = ${this.root.clientHeight}\\n` +\n        //   `this.host.style.height = ${this.host.style.height}\\n` +\n        //   `this.host.clientHeight = ${this.host.clientHeight}\\n` +\n        //   '\\n' +\n        //   `this.canvas.height       = ${this.canvas.height}\\n` +\n        //   `this.canvas.style.height = ${this.canvas.style.height}`);\n    }\n    /**\n     * @param {DG.Column} seqCol Column to look for a palette\n     * @param {number}  minLength minimum length of sequence to detect palette (empty strings are allowed)\n     * @return {SeqPalette} Palette corresponding to the alphabet of the sequences in the column\n     */\n    static pickUpPalette(seqCol, minLength = 5) {\n        let res = null;\n        switch (seqCol.semType) {\n            case Aminoacids.SemTypeMultipleAlignment:\n                res = AminoacidsPalettes.GrokGroups;\n                break;\n            case Nucleotides.SemTypeMultipleAlignment:\n                res = NucleotidesPalettes.Chromatogram;\n                break;\n        }\n        const stats = WebLogo.getStats(seqCol, minLength, WebLogo.splitterAsFasta);\n        const alphabetCandidates = [\n            [new Set(Object.keys(Nucleotides.Names)), NucleotidesPalettes.Chromatogram],\n            [new Set(Object.keys(Aminoacids.Names)), AminoacidsPalettes.GrokGroups],\n        ];\n        // Calculate likelihoods for alphabet_candidates\n        const alphabetCandidatesSim = alphabetCandidates\n            .map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[0]));\n        const maxCos = Math.max(...alphabetCandidatesSim);\n        if (maxCos > 0.55)\n            res = alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][1];\n        else\n            res = UnknownSeqPalettes.Color;\n        return res;\n    }\n    getAlphabetSize() {\n        var _a, _b;\n        return (_b = (_a = this.unitsHandler) === null || _a === void 0 ? void 0 : _a.getAlphabetSize()) !== null && _b !== void 0 ? _b : 0;\n    }\n    /** Stats of sequences with specified splitter func, returns { freq, sameLength }.\n     * @param {DG.Column} seqCol\n     * @param {number} minLength\n     * @param {SplitterFunc} splitter\n     * @return { SeqColStats }, sameLength: boolean } stats of column sequences\n     */\n    static getStats(seqCol, minLength, splitter) {\n        const freq = {};\n        let sameLength = true;\n        let firstLength = null;\n        for (const seq of seqCol.categories) {\n            const mSeq = splitter(seq);\n            if (firstLength == null)\n                firstLength = mSeq.length;\n            else if (mSeq.length !== firstLength)\n                sameLength = false;\n            if (mSeq.length > minLength) {\n                for (const m of mSeq) {\n                    if (!(m in freq))\n                        freq[m] = 0;\n                    freq[m] += 1;\n                }\n            }\n        }\n        return { freq: freq, sameLength: sameLength };\n    }\n    static getAlphabetSimilarity(freq, alphabet, gapSymbol = '-') {\n        const keys = new Set([...new Set(Object.keys(freq)), ...alphabet]);\n        keys.delete(gapSymbol);\n        const freqA = [];\n        const alphabetA = [];\n        for (const m of keys) {\n            freqA.push(m in freq ? freq[m] : 0);\n            alphabetA.push(alphabet.has(m) ? 1 : 0);\n        }\n        /* There were a few ideas: chi-squared, pearson correlation (variance?), scalar product */\n        const freqV = new Vector(freqA);\n        const alphabetV = new Vector(alphabetA);\n        return vectorDotProduct(freqV, alphabetV) / (vectorLength(freqV) * vectorLength(alphabetV));\n    }\n    // /** First try to find column with semType 'alignedSequence'.\n    //  * Next look for column with data alphabet corresponding to any of the known palettes.\n    //  * @param {DG.DataFrame} dataFrame\n    //  * @return {DG.Column} The column we were looking for or null\n    //  */\n    // private pickUpSeqCol(dataFrame: DG.DataFrame): DG.Column | null {\n    //   let res: DG.Column | null = dataFrame.columns.bySemType('alignedSequence');\n    //   if (res == null) {\n    //     for (const col of dataFrame.columns) {\n    //       const cp = WebLogo.pickUpPalette(col as DG.Column, 5);\n    //       if (cp !== null && !(cp instanceof UnknownSeqPalette)) {\n    //         res = col;\n    //         break;\n    //       }\n    //     }\n    //   }\n    //   return res;\n    // }\n    static pickUpSeqCol2(df) {\n        var _a;\n        const semTypeColList = df.columns.bySemTypeAll(DG.SEMTYPE.MACROMOLECULE);\n        let resCol = (_a = semTypeColList.find((col) => {\n            const units = col.getTag(DG.TAGS.UNITS);\n            return units ? units.indexOf('MSA') !== -1 : false;\n        })) !== null && _a !== void 0 ? _a : null;\n        if (!resCol && semTypeColList.length > 0)\n            resCol = semTypeColList[0];\n        return resCol;\n    }\n    /** Split sequence for single character monomers, square brackets multichar monomer names or gap symbol.\n     * @param {any} seq object with sequence\n     * @return {string[]} array of monomers\n     */\n    static splitterAsFasta(seq) {\n        const res = wu(seq.toString().matchAll(WebLogo.monomerRe))\n            .map((ma) => {\n            let mRes;\n            const m = ma[0];\n            if (m.length > 1) {\n                if (m in WebLogo.aaSynonyms) {\n                    mRes = WebLogo.aaSynonyms[m];\n                }\n                else {\n                    mRes = '';\n                    console.debug(`Long monomer '${m}' has not a short synonym.`);\n                }\n            }\n            else {\n                mRes = m;\n            }\n            return mRes;\n        }).toArray();\n        return res;\n    }\n    /** Splits Helm string to monomers, but does not replace monomer names to other notation (e.g. for RNA).\n     * Only for linear polymers, does not split RNA for ribose and phosphate monomers.\n     * @param {string} seq Source string of HELM notation\n     * @return {string[]}\n     */\n    static splitterAsHelm(seq) {\n        WebLogo.helmRe.lastIndex = 0;\n        const ea = WebLogo.helmRe.exec(seq.toString());\n        const inSeq = ea ? ea[2] : null;\n        const mmPostProcess = (mm) => {\n            WebLogo.helmPp1Re.lastIndex = 0;\n            const pp1M = WebLogo.helmPp1Re.exec(mm);\n            if (pp1M && pp1M.length >= 2)\n                return pp1M[1];\n            else\n                return mm;\n        };\n        const mmList = inSeq ? inSeq.split('.') : [];\n        const mmListRes = mmList.map(mmPostProcess);\n        return mmListRes;\n    }\n    /** Gets method to split sequence by separator\n     * @param {string} separator\n     * @param limit\n     * @return {SplitterFunc}\n     */\n    static getSplitterWithSeparator(separator, limit = undefined) {\n        return (seq) => {\n            return seq.split(separator, limit);\n        };\n    }\n    /** Get splitter method to split sequences to monomers\n     * @param {string} units\n     * @param {string} separator\n     * @param limit\n     * @return {SplitterFunc}\n     */\n    static getSplitter(units, separator, limit = undefined) {\n        if (units.toLowerCase().startsWith(\"fasta\" /* NOTATION.FASTA */))\n            return WebLogo.splitterAsFasta;\n        else if (units.toLowerCase().startsWith(\"separator\" /* NOTATION.SEPARATOR */))\n            return WebLogo.getSplitterWithSeparator(separator, limit);\n        else if (units.toLowerCase().startsWith(\"helm\" /* NOTATION.HELM */))\n            return WebLogo.splitterAsHelm;\n        else\n            throw new Error(`Unexpected units ${units} .`);\n        // TODO: Splitter for HELM\n    }\n    static getSplitterForColumn(col) {\n        if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n            throw new Error(`Get splitter for semType \"${DG.SEMTYPE.MACROMOLECULE}\" only.`);\n        const units = col.getTag(DG.TAGS.UNITS);\n        const separator = col.getTag(UnitsHandler.TAGS.separator);\n        return WebLogo.getSplitter(units, separator);\n    }\n    /* Shortens monomer representation text for monomers with long names.\n    **/\n    static monomerToText(src) {\n        const srcTxt = src.toString();\n        if (srcTxt.length <= 5) {\n            return srcTxt;\n        }\n        else {\n            const parts = wu(src.toString().matchAll(WebLogo.longMonomerPartRe))\n                .map((ma) => {\n                const mRes = ma[0];\n                return mRes;\n            }).toArray();\n            if (parts.length == 0) {\n                return ' ';\n            }\n            else {\n                const part0 = parts[0];\n                const resTxt = part0.length < 6 ? `${part0}…` : `${part0.substring(0, 5)}…`;\n                return resTxt;\n            }\n        }\n    }\n    static monomerToShort(amino, maxLengthOfMonomer) {\n        return amino.length <= maxLengthOfMonomer ? amino : amino.substring(0, maxLengthOfMonomer) + '…';\n    }\n}\nWebLogo.residuesSet = 'nucleotides';\nWebLogo.viewerCount = -1;\nWebLogo.monomerRe = /\\[(\\w+)\\]|(\\w)|(-)/g;\nWebLogo.helmRe = /(PEPTIDE1|DNA1|RNA1)\\{([^}]+)}/g;\nWebLogo.helmPp1Re = /\\[([^\\[\\]]+)]/g;\n/** Only some of the synonyms. These were obtained from the clustered oligopeptide dataset. */\nWebLogo.aaSynonyms = {\n    '[MeNle]': 'L',\n    '[MeA]': 'A',\n    '[MeG]': 'G',\n    '[MeF]': 'F',\n};\nWebLogo.longMonomerPartRe = /(\\w+)/g;\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"web-logo.js","sourceRoot":"","sources":["web-logo.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,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAE7B,OAAO,EAAC,MAAM,EAAC,MAAM,iDAAiD,CAAC;AACvE,OAAO,EAAC,YAAY,EAAE,gBAAgB,EAAC,MAAM,iDAAiD,CAAC;AAC/F,OAAO,EAAC,UAAU,EAAE,kBAAkB,EAAC,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAC,WAAW,EAAE,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAC,kBAAkB,EAAC,MAAM,YAAY,CAAC;AAG9C,OAAO,EAAW,YAAY,EAAC,MAAM,wBAAwB,CAAC;AAQ9D,IAAK,cAGJ;AAHD,WAAK,cAAc;IACjB,qCAAmB,CAAA;IACnB,+BAAa,CAAA;AACf,CAAC,EAHI,cAAc,KAAd,cAAc,QAGlB;AAYD;;;GAGG;AACH,iBAAiB,CAAC,SAAS,CAAC,iBAAiB,GAAG,UAAS,KAAiB,EAAE,CAAS;IACnF,MAAM,IAAI,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC1C,OAAO,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;AACvF,CAAC,CAAC;AAEF,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,GAAG,UAAS,CAAS,EAAE,CAAS;IACxD,OAAO,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC;AAChF,CAAC,CAAC;AAEF,MAAM,OAAO,mBAAmB;IAS9B,YAAY,QAAgB,CAAC,EAAE,SAAkB,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;CACF;AAED,MAAM,OAAO,YAAY;IAMvB;;;;;OAKG;IACH,YAAY,IAAY,EAAE,OAA6C,EAAE,EAAE,WAAmB,CAAC,EAAE,mBAA2B,CAAC;QAC3H,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,gBAAgB,GAAG,gBAAgB,CAAC;IAC3C,CAAC;CACF;AAED,MAAM,OAAO,OAAQ,SAAQ,EAAE,CAAC,QAAQ;IAyEtC;QACE,KAAK,EAAE,CAAC;QAtEF,aAAQ,GAAW,CAAC,CAAC,CAAC;QAEtB,gBAAW,GAAY,KAAK,CAAC;QAErC,4FAA4F;QAClF,OAAE,GAAsB,IAAI,CAAC;QAQ/B,eAAU,GAAW,EAAE,CAAC;QAExB,WAAM,GAA6B,IAAI,CAAC;QACxC,aAAQ,GAAwB,IAAI,CAAC;QAC7C,mCAAmC;QAC3B,cAAS,GAAmB,EAAE,CAAC;QAE/B,eAAU,GAAW,CAAC,CAAC;QACvB,aAAQ,GAAW,CAAC,CAAC;QAUtB,mBAAc,GAAW,CAAC,CAAC;QAW1B,kBAAa,GAAa,EAAE,CAAC;QAE7B,kBAAa,GAAW,CAAC,CAAC,CAAC;QAE3B,gBAAW,GAAW,CAAC,CAAC,CAAC;QAqBzB,aAAQ,GAAmB,EAAE,CAAC;QAKpC,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC;QACpC,OAAO,CAAC,WAAW,IAAI,CAAC,CAAC;QAEzB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QAEzB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,EAAE,EAAE,CAAA;8DACnB,CAAC,CAAC;QACxD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,EAAE,CAAA;gEACa,CAAC,CAAC;QAC1D,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,GAAG,CAAA;gEACY,CAAC,CAAC;QAE1D,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,IAAI,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;QACvE,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAC;QAElE,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAChE,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC,CAAC;QAE5D,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;QAE7C,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,mBAAmB,EAAE,QAAQ,EAChE,EAAC,OAAO,EAAE,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,EAAC,CAAC,CAAC;QAC1C,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,MAAM,CAAC,qBAAqB,EAAE,QAAQ,EACpE,EAAC,OAAO,EAAE,CAAC,MAAM,EAAE,QAAQ,EAAE,OAAO,CAAC,EAAC,CAAC,CAAC;QAC1C,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,CAAC;QAC1D,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,KAAK,CAAC,CAAC;QACjE,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,MAAM,CAAC,qBAAqB,EAAE,MAAM,EAClE,EAAC,OAAO,EAAE,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,EAAC,CAAC,CAAC;QACxC,IAAI,6BAA6B,GAAG,CAAC,CAAC;QACtC,IAAI,IAAI,CAAC,mBAAmB,KAAK,MAAM,EAAE;YACrC,6BAA6B,GAAG,CAAC,CAAC;SACrC;QACD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC,gBAAgB,EAAE,6BAA6B,EAAE,EAAC,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,EAAE,EAAC,CAAC,CAAC;QACnG,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,gBAAgB,EAAE,cAAc,CAAC,IAAI,EAAE,EAAC,OAAO,EAAE,CAAC,cAAc,CAAC,IAAI,EAAE,cAAc,CAAC,OAAO,CAAC,EAAC,CAAC,CAAC;IACrI,CAAC;IA5DD,0DAA0D;IAC1D,IAAY,MAAM;QAChB,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;SAC9B;QACD,OAAO,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAChG,CAAC;IAED,IAAY,uBAAuB;;QACjC,IAAI,CAAC,IAAI,CAAC,mBAAmB,KAAK,MAAM,CAAC,IAAI,CAAC,CAAA,MAAA,IAAI,CAAC,YAAY,0CAAE,sBAAsB,EAAE,MAAK,IAAI,CAAC,EAAE;YACnG,OAAO,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;SAClD;QACD,IAAI,IAAI,CAAC,mBAAmB,KAAK,QAAQ,EAAE;YACzC,OAAO,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;SAClD;QAED,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;IA6Ca,IAAI;;YAChB,IAAI,IAAI,CAAC,WAAW,EAAE;gBACpB,OAAO,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;gBAChD,OAAO;aACR;YAED,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;YACxB,IAAI,CAAC,OAAO,GAAG,qCAAqC,CAAC;YAErD,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YACpC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;YAEpC,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;YAC1B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YAEjC,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;YAEhD,6CAA6C;YAC7C,yCAAyC;YACzC,0CAA0C;YAE1C,MAAM,UAAU,GAAG,CAAC,CAAW,EAAuD,EAAE;gBACtF,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,CAAC;gBAC5D,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBAEtC,IAAI,QAAQ,KAAK,KAAK,CAAC;oBACrB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,MAAM,OAAO,GAAuB,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;qBAC3D,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAC3D,IAAI,OAAO,KAAK,SAAS;oBACvB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBAE5B,OAAO,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;YACjD,CAAC,CAAC;YAGF,qGAAqG;YACrG,OAAO;YAEP,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAa,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAa,EAAE,EAAE;gBAClG,IAAI,CAAC,IAAI,CAAC,MAAM;oBACd,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAE7B,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;gBAC1C,MAAM,OAAO,GAAa,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;gBACjE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;gBAE5C,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,IAAI,OAAO,EAAE;oBAC7D,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,CAAC,IAAY,EAAE,EAAE;wBAChF,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzE,OAAO,CAAC,CAAC,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,IAAI,KAAK,EAAE,IAAI,OAAO,KAAK,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;oBACrG,CAAC,CAAC,CAAC,MAAM,CAAS,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;oBACjD,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,OAAO,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,QAAQ,OAAO,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;iBACvG;qBAAM;oBACL,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;iBACnB;YACH,CAAC,CAAC,CAAC,CAAC;YAEJ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAa,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC,CAAa,EAAE,EAAE;gBAClG,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC;oBAC/B,OAAO;gBAET,MAAM,IAAI,GAAG,CAAe,CAAC;gBAC7B,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;gBAC1C,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;gBAE3E,uDAAuD;gBACvD,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,IAAI,OAAO,EAAE;oBAC7D,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE;wBACrC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;wBACnC,MAAM,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;wBACzE,OAAO,CAAC,CAAC,IAAI,KAAK,OAAO,CAAC,IAAI,CAAC,IAAI,KAAK,EAAE,IAAI,OAAO,KAAK,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;oBACrG,CAAC,CAAC,CAAC;iBACJ;YACH,CAAC,CAAC,CAAC,CAAC;YAEJ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAE7F,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC5B,2CAA2C;YAE3C,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACpB,CAAC;KAAA;IAEO,iBAAiB,CAAC,IAAS;QACjC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAElB,iDAAiD;QACjD,qFAAqF;QACrF,iFAAiF;IACnF,CAAC;IAED;OACG;IACK,YAAY;QAClB,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YAC3F,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,EAAE;gBACvB,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACpD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;aACjE;YACD,IAAI,IAAI,CAAC,MAAM,EAAE;gBACf,MAAM,KAAK,GAAW,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACzD,MAAM,SAAS,GAAW,IAAI,CAAC,MAAO,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBAC3E,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;gBACtD,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAElD,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,EAAE,GAAG,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;aAC9C;iBAAM;gBACL,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;gBACrB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;gBACxB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;gBACxB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC;gBACtB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;aAChB;SACF;QACD,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAEO,eAAe;QACrB,IAAI,CAAC,IAAI,CAAC,MAAM;YACd,OAAO;QAET,IAAI,UAA8B,CAAC;QACnC,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,MAAM,OAAO,GAAe,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACvE,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAC7B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAS,EAAE,EAAE,CAAC,IAAI,CAAC,MAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SAC/F;aAAM;YACL,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC;SACrC;QACD,MAAM,SAAS,GAAG,UAAU,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,GAAG,CAClE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,QAAS,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzD,0DAA0D;QAC1D,MAAM,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAC7D,kDAAkD;QAClD,IAAI,CAAC,aAAa,GAAG,gBAAgB,CAAC,CAAC,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACzF,CAAC,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,EAAE,CAAC,CAAC;QAE5D,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,aAAa;YAChE,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;YACtD,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACzD,IAAI,CAAC,WAAW,GAAG,CAAC,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,aAAa;YAC5D,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;YACpD,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;IACvE,CAAC;IAEe,iBAAiB,CAAC,QAAqB;QACrD,wEAAwE;QACxE,KAAK,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;QAElC,QAAQ,QAAQ,CAAC,IAAI,EAAE;YACvB,KAAK,uBAAuB;gBAC1B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,oBAAoB;gBACvB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,mBAAmB;gBACtB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,iBAAiB;gBACpB,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM;YACR,KAAK,eAAe;gBAClB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC;gBACzC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,UAAU;gBACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,mBAAmB;gBACtB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,qBAAqB;gBACxB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,SAAS;gBACZ,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,iBAAiB;gBACpB,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,oBAAoB;gBACvB,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,gBAAgB;gBACnB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,qBAAqB;gBACxB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;YACR,KAAK,gBAAgB;gBACnB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAClB,MAAM;SACP;IACH,CAAC;IAEqB,eAAe;;;;;YACnC,MAAM,YAAY,GAAW,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;YAC9D,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,kCAAkC,YAAY,UAAU,CAAC,CAAC;YACrG,OAAM,eAAe,YAAG;YAExB,IAAI,CAAC,YAAY,EAAE,CAAC;YAEpB,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,CAAC,EAAE;gBAC7B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBACnF,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;oBAC7D,IAAI,CAAC,eAAe,EAAE,CAAC;oBACvB,IAAI,CAAC,MAAM,EAAE,CAAC;gBAChB,CAAC,CAAC,CAAC,CAAC;aACL;YAED,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,yBAAyB,CAAC,CAAC;QACxE,CAAC;KAAA;IAEqB,MAAM;;;;;YAC1B,MAAM,YAAY,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;YAC3D,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,kCAAkC,YAAY,UAAU,CAAC,CAAC;YACrG,OAAM,MAAM,YAAG;YAEf,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;YAClD,IAAI,CAAC,IAAK,CAAC,MAAM,EAAE,CAAC;YACpB,IAAI,CAAC,MAAM,GAAG,SAAS,CAAC;YACxB,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;YACzB,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC;YAEtB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACzB,OAAO,CAAC,KAAK,CAAC,gBAAgB,IAAI,CAAC,QAAQ,yBAAyB,CAAC,CAAC;QACxE,CAAC;KAAA;IAES,aAAa,CAAC,aAAa,GAAG,GAAG;QACzC,IAAI,IAAI,CAAC,qBAAqB;YAC5B,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAE7D,OAAO,EAAE,CAAC;IACZ,CAAC;IAES,WAAW,CAAC,KAAiB,EAAE,SAA8B;QACrE,IAAI,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC1B,IAAI,cAAc,GAAG,CAAC,CAAC;QACvB,KAAK,IAAI,cAAc,GAAG,CAAC,EAAE,cAAc,GAAG,MAAM,EAAE,cAAc,EAAE,EAAE;YACtE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC,EAAE;gBACrC,KAAK,CAAC,cAAc,CAAC,GAAG,KAAK,CAAC,cAAc,CAAC,CAAC;gBAC9C,cAAc,EAAE,CAAC;aAClB;SACF;QACD,KAAK,CAAC,MAAM,GAAG,cAAc,CAAC;QAC9B,OAAO,KAAK,CAAC;IACf,CAAC;IAGS,qBAAqB;QAC7B,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,EAAE,WAAC,OAAA,CAAA,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,IAAI,CAAC,GAAG,CAAC,0CAAE,KAAK,MAAK,IAAI,CAAC,QAAQ,CAAA,EAAA,CAAC,CAAC;SACpF;IACH,CAAC;IAES,UAAU,CAAC,CAAS;QAC5B,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS;YAC/D,OAAO;QACT,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAGlD,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEhB,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,CAAC,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnH,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,OAAO,GAAW,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC;YACtE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,YAAY,CAAC,OAAO,CAAC,CAAC;SAClD;QAED,iFAAiF;QACjF,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;QAC3D,2GAA2G;QAC3G,gDAAgD;QAEhD,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC;QACjC,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;QAElB,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE;YACvB,IAAI,CAAC,GAAmB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YAE7C,IAAI,CAAC,CAAC,EAAE;gBACN,CAAC,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;gBACzB,EAAE,IAAI,CAAC,QAAQ,CAAC;aACjB;YAED,MAAM,IAAI,GAAa,IAAI,CAAC,QAAS,CAAC,CAAC,CAAC,CAAC;YACzC,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;gBAC7C,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;gBACzC,MAAM,CAAC,GAAW,IAAI,CAAC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,IAAI,GAAG,CAAC;gBACzD,IAAI,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC;oBAChB,MAAM,CAAC,CAAC,CAAC,GAAG,IAAI,mBAAmB,EAAE,CAAC;gBACxC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC;aACnB;SACF;QAED,4BAA4B;QAC5B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,yCAAyC;YAEzC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,GAAG,CAAC,CAAC;YAClC,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI;gBACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YACtE,IAAI,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,EAAE;gBACjD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,GAAG,CAAC,CAAC;gBAC1C,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE;oBACzC,MAAM,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC;oBAC9E,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;iBAC9D;aACF;SACF;QACD,YAAY;QACZ,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAE7B,MAAM,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;QACnE,mFAAmF;QAEnF,0BAA0B;QAC1B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,IAAI,GAAyC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC;YAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC;YAC/C,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;YAC5C,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,EAAE;gBAC7E,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;aACrD;YAED,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,gBAAgB,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;YAE5M,IAAI,CAAC,GAAW,IAAI,CAAC,UAAU,GAAG,CAAC,GAAG,CAAC,iBAAiB,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC;YAE1E,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBACjD,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBAC9B,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;qBAC5B,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnC,OAAO,CAAC,CAAC;qBACN,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG;oBACnB,OAAO,CAAC,CAAC,CAAC;qBACP,oBAAoB;oBACvB,OAAO,CAAC,CAAC,CAAC;YACd,CAAC,CAAC,CAAC;YACH,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE;gBAC3B,MAAM,MAAM,GAAwB,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC7C,8BAA8B;gBAC9B,MAAM,CAAC,GAAW,SAAS,GAAG,MAAM,CAAC,KAAK,GAAG,QAAQ,CAAC;gBAEtD,MAAM,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC,EAAE,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;gBAC5F,CAAC,IAAI,CAAC,CAAC;aACR;SACF;QACD,YAAY;IACd,CAAC;IAED,2CAA2C;IAC3C,MAAM,CAAC,MAAM,GAAG,IAAI;;QAClB,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;gBAC3B,IAAI,CAAC,OAAQ,CAAC,SAAS,GAAG,qCAAqC,IAAI,CAAC,MAAM,CAAC,OAAO,KAAK,CAAC;gBACxF,IAAI,CAAC,OAAQ,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC;aAClC;iBAAM;gBACL,IAAI,CAAC,OAAQ,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;aACtC;SACF;QAED,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,EAAE;YAC7D,OAAO;QAET,MAAM,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC,CAAC;YAAE,OAAO;QAEf,MAAM,CAAC,GAAG,MAAM,CAAC,gBAAgB,CAAC;QAElC,IAAI,MAAM;YACR,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAErB,CAAC,CAAC,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC;QACtB,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACxD,CAAC,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QAEnC,0CAA0C;QAC1C,IAAI,IAAI,CAAC,aAAa,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,KAAK,CAAC,CAAC;YACtD,OAAO;QAET,4BAA4B;QAC5B,MAAM,gBAAgB,GAAG,EAAE,GAAG,CAAC,CAAC;QAChC,CAAC,CAAC,cAAc,EAAE,CAAC;QACnB,CAAC,CAAC,SAAS,GAAG,OAAO,CAAC;QACtB,CAAC,CAAC,SAAS,GAAG,QAAQ,CAAC;QACvB,CAAC,CAAC,IAAI,GAAG,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC,CAAC,qCAAqC,CAAC;QAC7E,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QAChG,MAAM,MAAM,GAAG,eAAe,GAAG,CAAC,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC,GAAG,eAAe,CAAC;QAE7G,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,MAAM,GAAG,GAAiB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC/C,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,CAAC,CAAC,YAAY,CACZ,MAAM,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EACf,IAAI,GAAG,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,cAAc,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YACpE,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;SAC5B;QACD,+BAA+B;QAE/B,KAAK,IAAI,IAAI,GAAG,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,EAAE;YAC7C,KAAK,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE;gBACzE,IAAI,OAAO,KAAK,GAAG,EAAE;oBACnB,MAAM,UAAU,GAAG,OAAO,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;oBAClD,MAAM,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;oBAExB,MAAM,SAAS,GAAG,uCAAuC,CAAC;oBAC1D,0DAA0D;oBAC1D,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBACnC,MAAM,qBAAqB,GAAG,IAAI,CAAC;oBAEnC,CAAC,CAAC,cAAc,EAAE,CAAC;oBACnB,CAAC,CAAC,WAAW,GAAG,WAAW,CAAC;oBAC5B,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC;oBAChB,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC;oBACzC,CAAC,CAAC,SAAS,GAAG,MAAA,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,mCAAI,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;oBAC3D,CAAC,CAAC,SAAS,GAAG,MAAM,CAAC;oBACrB,CAAC,CAAC,IAAI,GAAG,SAAS,CAAC;oBACnB,+CAA+C;oBAC/C,MAAM,GAAG,GAAgB,CAAC,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;oBAEnD,uCAAuC;oBACvC,yFAAyF;oBAEzF,CAAC,CAAC,YAAY,CACZ,CAAC,CAAC,KAAK,GAAG,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,MAAM,GAAG,qBAAqB,EAC3D,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC;oBACjB,CAAC,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC,EAAE,CAAC,qBAAqB,CAAC,CAAC;iBACnD;aACF;SACF;IACH,CAAC;IAEO,QAAQ;QACd,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI;YAC5B,OAAO;QAET,MAAM,CAAC,GAAW,MAAM,CAAC,gBAAgB,CAAC;QAE1C,IAAI,KAAK,GAAW,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,uBAAuB,GAAG,CAAC,CAAC;QACnE,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;QAExF,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,MAAM,MAAM,GAAW,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC;YACvD,MAAM,MAAM,GAAW,IAAI,CAAC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YACrD,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC;YACpD,KAAK,GAAG,KAAK,GAAG,KAAK,CAAC;YACtB,MAAM,GAAG,MAAM,GAAG,KAAK,CAAC;YACxB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;SAClD;QAED,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,GAAG,CAAC,CAAC;QAC9B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAC;QAEvC,oFAAoF;QACpF,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,MAAM,IAAI,CAAC,CAAC;QACrD,MAAM,YAAY,GAAW,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC;QACpD,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,YAAY,GAAG,CAAC,CAAC;QACtC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,YAAY,IAAI,CAAC,CAAC;QAE7D,6BAA6B;QAC7B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,sCAAsC;YACtC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAC;YAC7D,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC;YACvC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,UAAU,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;SAChE;aAAM;YACL,8BAA8B;YAC9B,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC;YACvD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,gBAAgB,CAAC;YAC7C,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC;YAE9C,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,IAAI,CAAC,mBAAmB,CAAC,CAAC;YAEpE,qBAAqB;YACrB,IAAI,aAAa,GAAG,CAAC,CAAC;YACtB,QAAQ,IAAI,CAAC,iBAAiB,EAAE;gBAChC,KAAK,KAAK;oBACR,aAAa,GAAG,CAAC,CAAC;oBAClB,MAAM;gBACR,KAAK,QAAQ;oBACX,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;oBACnE,MAAM;gBACR,KAAK,QAAQ;oBACX,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,CAAC;oBAC7D,MAAM;aACP;YACD,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,GAAG,aAAa,IAAI,EAAE,WAAW,CAAC,CAAC;YAE7E,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,MAAM,EAAE;gBACnC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,CAAC;gBACrE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;aACjD;iBAAM;gBACL,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;aAClE;SACF;QAED,iBAAiB;QACjB,6DAA6D;QAC7D,6DAA6D;QAC7D,6DAA6D;QAC7D,6DAA6D;QAC7D,WAAW;QACX,2DAA2D;QAC3D,+DAA+D;IACjE,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,aAAa,CAAC,MAAiB,EAAE,YAAoB,CAAC;QAClE,IAAI,GAAG,GAAsB,IAAI,CAAC;QAClC,QAAQ,MAAM,CAAC,OAAO,EAAE;YACxB,KAAK,UAAU,CAAC,wBAAwB;gBACtC,GAAG,GAAG,kBAAkB,CAAC,UAAU,CAAC;gBACpC,MAAM;YACR,KAAK,WAAW,CAAC,wBAAwB;gBACvC,GAAG,GAAG,mBAAmB,CAAC,YAAY,CAAC;gBACvC,MAAM;SACP;QACD,MAAM,KAAK,GAAgB,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC;QAExF,MAAM,kBAAkB,GAAgC;YACtD,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,mBAAmB,CAAC,YAAY,CAAC;YAC3E,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,kBAAkB,CAAC,UAAU,CAAC;SACxE,CAAC;QACF,gDAAgD;QAChD,MAAM,qBAAqB,GAAa,kBAAkB;aACvD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,qBAAqB,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,qBAAqB,CAAC,CAAC;QAClD,IAAI,MAAM,GAAG,IAAI;YACf,GAAG,GAAG,kBAAkB,CAAC,qBAAqB,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;;YAEnE,GAAG,GAAG,kBAAkB,CAAC,KAAK,CAAC;QAEjC,OAAO,GAAG,CAAC;IACb,CAAC;IAEM,eAAe;;QACpB,OAAO,MAAA,MAAA,IAAI,CAAC,YAAY,0CAAE,eAAe,EAAE,mCAAI,CAAC,CAAC;IACnD,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,QAAQ,CAAC,MAAiB,EAAE,SAAiB,EAAE,QAAsB;QAC1E,MAAM,IAAI,GAA4B,EAAE,CAAC;QACzC,IAAI,UAAU,GAAG,IAAI,CAAC;QACtB,IAAI,WAAW,GAAG,IAAI,CAAC;QAEvB,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,UAAU,EAAE;YACnC,MAAM,IAAI,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;YAE3B,IAAI,WAAW,IAAI,IAAI;gBACrB,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC;iBACvB,IAAI,IAAI,CAAC,MAAM,KAAK,WAAW;gBAClC,UAAU,GAAG,KAAK,CAAC;YAErB,IAAI,IAAI,CAAC,MAAM,GAAG,SAAS,EAAE;gBAC3B,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE;oBACpB,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;wBACd,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;oBACd,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;iBACd;aACF;SACF;QACD,OAAO,EAAC,IAAI,EAAE,IAAI,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC;IAC9C,CAAC;IAEM,MAAM,CAAC,qBAAqB,CAAC,IAAkB,EAAE,QAAqB,EAAE,YAAoB,GAAG;QACpG,MAAM,IAAI,GAAG,IAAI,GAAG,CAAS,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;QAC3E,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAEvB,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,MAAM,SAAS,GAAa,EAAE,CAAC;QAC/B,KAAK,MAAM,CAAC,IAAI,IAAI,EAAE;YACpB,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACpC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SACzC;QACD,0FAA0F;QAC1F,MAAM,KAAK,GAAW,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;QACxC,MAAM,SAAS,GAAW,IAAI,MAAM,CAAC,SAAS,CAAC,CAAC;QAChD,OAAO,gBAAgB,CAAC,KAAK,EAAE,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC;IAC9F,CAAC;IAED,+DAA+D;IAC/D,yFAAyF;IACzF,qCAAqC;IACrC,gEAAgE;IAChE,MAAM;IACN,oEAAoE;IACpE,gFAAgF;IAChF,uBAAuB;IACvB,6CAA6C;IAC7C,+DAA+D;IAC/D,iEAAiE;IACjE,qBAAqB;IACrB,iBAAiB;IACjB,UAAU;IACV,QAAQ;IACR,MAAM;IACN,gBAAgB;IAChB,IAAI;IAEG,MAAM,CAAC,aAAa,CAAC,EAAgB;;QAC1C,MAAM,cAAc,GAAG,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QACzE,IAAI,MAAM,GAAqB,MAAA,cAAc,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;YACzD,MAAM,KAAK,GAAG,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACxC,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QACrD,CAAC,CAAC,mCAAI,IAAI,CAAC;QACX,IAAI,CAAC,MAAM,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC;YACtC,MAAM,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;QAC7B,OAAO,MAAM,CAAC;IAChB,CAAC;IAID;;;OAGG;IACI,MAAM,CAAC,eAAe,CAAC,GAAQ;QACpC,MAAM,GAAG,GAAa,EAAE,CAAmB,GAAG,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;aACnF,GAAG,CAAC,CAAC,EAAoB,EAAE,EAAE;YAC5B,IAAI,IAAY,CAAC;YACjB,MAAM,CAAC,GAAW,EAAE,CAAC,CAAC,CAAC,CAAC;YACxB,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;gBAChB,IAAI,CAAC,IAAI,OAAO,CAAC,UAAU,EAAE;oBAC3B,IAAI,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;iBAC9B;qBAAM;oBACL,IAAI,GAAG,EAAE,CAAC;oBACV,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,4BAA4B,CAAC,CAAC;iBAC/D;aACF;iBAAM;gBACL,IAAI,GAAG,CAAC,CAAC;aACV;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;QAEf,OAAO,GAAG,CAAC;IACb,CAAC;IAKD;;;;OAIG;IACI,MAAM,CAAC,cAAc,CAAC,GAAQ;QACnC,OAAO,CAAC,MAAM,CAAC,SAAS,GAAG,CAAC,CAAC;QAC7B,MAAM,EAAE,GAA2B,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;QACvE,MAAM,KAAK,GAAkB,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAE/C,MAAM,aAAa,GAAG,CAAC,EAAU,EAAU,EAAE;YAC3C,OAAO,CAAC,SAAS,CAAC,SAAS,GAAG,CAAC,CAAC;YAChC,MAAM,IAAI,GAAG,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACxC,IAAI,IAAI,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC;gBAC1B,OAAO,IAAI,CAAC,CAAC,CAAC,CAAC;;gBAEf,OAAO,EAAE,CAAC;QACd,CAAC,CAAC;QAEF,MAAM,MAAM,GAAa,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACvD,MAAM,SAAS,GAAa,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;QACtD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,wBAAwB,CAAC,SAAiB,EAAE,QAA4B,SAAS;QAC7F,OAAO,CAAC,GAAW,EAAE,EAAE;YACrB,OAAO,GAAG,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;QACrC,CAAC,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACI,MAAM,CAAC,WAAW,CAAC,KAAa,EAAE,SAAiB,EAAE,QAA4B,SAAS;QAC/F,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,8BAAgB;YAChD,OAAO,OAAO,CAAC,eAAe,CAAC;aAC5B,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,sCAAoB;YACzD,OAAO,OAAO,CAAC,wBAAwB,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;aACvD,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC,UAAU,4BAAe;YACpD,OAAO,OAAO,CAAC,cAAc,CAAC;;YAE9B,MAAM,IAAI,KAAK,CAAC,oBAAoB,KAAK,IAAI,CAAC,CAAC;QAEjD,0BAA0B;IAC5B,CAAC;IAEM,MAAM,CAAC,oBAAoB,CAAC,GAAc;QAC/C,IAAI,GAAG,CAAC,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC,aAAa;YAC1C,MAAM,IAAI,KAAK,CAAC,6BAA6B,EAAE,CAAC,OAAO,CAAC,aAAa,SAAS,CAAC,CAAC;QAElF,MAAM,KAAK,GAAG,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxC,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1D,OAAO,OAAO,CAAC,WAAW,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IAC/C,CAAC;IAYD;OACG;IACI,MAAM,CAAC,aAAa,CAAC,GAAQ;QAClC,MAAM,MAAM,GAAW,GAAG,CAAC,QAAQ,EAAE,CAAC;QACtC,IAAI,MAAM,CAAC,MAAM,IAAI,CAAC,EAAE;YACtB,OAAO,MAAM,CAAC;SACf;aAAM;YACL,MAAM,KAAK,GAAa,EAAE,CAAmB,GAAG,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;iBAC7F,GAAG,CAAC,CAAC,EAAoB,EAAE,EAAE;gBAC5B,MAAM,IAAI,GAAW,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC3B,OAAO,IAAI,CAAC;YACd,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;YAEf,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,EAAE;gBACrB,OAAO,GAAG,CAAC;aACZ;iBAAM;gBACL,MAAM,KAAK,GAAW,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC/B,MAAM,MAAM,GAAW,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC;gBACpF,OAAO,MAAM,CAAC;aACf;SACF;IACH,CAAC;IAEM,MAAM,CAAC,cAAc,CAAC,KAAa,EAAE,kBAA0B;QACpE,OAAO,KAAK,CAAC,MAAM,IAAI,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,kBAAkB,CAAC,GAAG,GAAG,CAAC;IACnG,CAAC;;AAh3Ba,mBAAW,GAAG,aAAa,CAAC;AAC3B,mBAAW,GAAW,CAAC,CAAC,CAAC;AA+uBzB,iBAAS,GAAG,qBAAqB,CAAC;AA2BlC,cAAM,GAAG,iCAAiC,CAAC;AAC3C,iBAAS,GAAG,gBAAgB,CAAC;AAiE5C,8FAA8F;AAC/E,kBAAU,GAA+B;IACtD,SAAS,EAAE,GAAG;IACd,OAAO,EAAE,GAAG;IACZ,OAAO,EAAE,GAAG;IACZ,OAAO,EAAE,GAAG;CACb,CAAC;AAEa,yBAAiB,GAAG,QAAQ,CAAC","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\n\nimport wu from 'wu';\nimport * as rxjs from 'rxjs';\n\nimport {Vector} from '@datagrok-libraries/utils/src/type-declarations';\nimport {vectorLength, vectorDotProduct} from '@datagrok-libraries/utils/src/vector-operations';\nimport {Aminoacids, AminoacidsPalettes} from '../aminoacids';\nimport {Nucleotides, NucleotidesPalettes} from '../nucleotides';\nimport {UnknownSeqPalettes} from '../unknown';\nimport {SeqPalette} from '../seq-palettes';\nimport {Subscription} from 'rxjs';\nimport {NOTATION, UnitsHandler} from '../utils/units-handler';\n\ndeclare module 'datagrok-api/src/grid' {\n  interface Rect {\n    contains(x: number, y: number): boolean;\n  }\n}\n\nenum PositionHeight {\n  Entropy = 'Entropy',\n  full = '100%',\n}\n\ndeclare global {\n  interface HTMLCanvasElement {\n    getCursorPosition(event: MouseEvent, r: number): DG.Point;\n  }\n}\n\nexport type MonomerFreqs = { [m: string]: number };\nexport type SeqColStats = { freq: MonomerFreqs, sameLength: boolean }\nexport type SplitterFunc = (seq: string) => string[];\n\n/**@param {MouseEvent} event\n * @param {number} r devicePixelRation\n * @return {DG.Point} canvas related cursor position\n */\nHTMLCanvasElement.prototype.getCursorPosition = function(event: MouseEvent, r: number): DG.Point {\n  const rect = this.getBoundingClientRect();\n  return new DG.Point((event.clientX - rect.left) * r, (event.clientY - rect.top) * r);\n};\n\nDG.Rect.prototype.contains = function(x: number, y: number): boolean {\n  return this.left <= x && x <= this.right && this.top <= y && y <= this.bottom;\n};\n\nexport class PositionMonomerInfo {\n  /** Sequences count with monomer in position\n   */\n  count: number;\n\n  /** Remember screen coords rect\n   */\n  bounds: DG.Rect;\n\n  constructor(count: number = 0, bounds: DG.Rect = new DG.Rect(0, 0, 0, 0)) {\n    this.count = count;\n    this.bounds = bounds;\n  }\n}\n\nexport class PositionInfo {\n  public readonly name: string;\n  freq: { [m: string]: PositionMonomerInfo };\n  rowCount: number;\n  sumForHeightCalc: number;\n\n  /** freq = {}, rowCount = 0\n   * @param {string} name Name of position ('111A', '111.1', etc)\n   * @param {number} sumForHeightCalc Sum of all monomer counts for height calculation\n   * @param {number} rowCount Count of elements in column\n   * @param {string[]} freq frequency of monomers in position\n   */\n  constructor(name: string, freq: { [m: string]: PositionMonomerInfo } = {}, rowCount: number = 0, sumForHeightCalc: number = 0) {\n    this.name = name;\n    this.freq = freq;\n    this.rowCount = rowCount;\n    this.sumForHeightCalc = sumForHeightCalc;\n  }\n}\n\nexport class WebLogo extends DG.JsViewer {\n  public static residuesSet = 'nucleotides';\n  private static viewerCount: number = -1;\n\n  private viewerId: number = -1;\n  private unitsHandler: UnitsHandler | null;\n  private initialized: boolean = false;\n\n  // private readonly colorScheme: ColorScheme = ColorSchemes[NucleotidesWebLogo.residuesSet];\n  protected cp: SeqPalette | null = null;\n\n  private host?: HTMLDivElement;\n  private msgHost?: HTMLElement;\n  private canvas?: HTMLCanvasElement;\n  private slider?: DG.RangeSlider;\n  private textBaseline: CanvasTextBaseline;\n\n  private axisHeight: number = 12;\n\n  private seqCol: DG.Column<string> | null = null;\n  private splitter: SplitterFunc | null = null;\n  // private maxLength: number = 100;\n  private positions: PositionInfo[] = [];\n\n  private rowsMasked: number = 0;\n  private rowsNull: number = 0;\n\n  // Viewer's properties (likely they should be public so that they can be set outside)\n  private _positionWidth: number;\n  public positionWidth: number;\n  public minHeight: number;\n  public maxHeight: number;\n  public considerNullSequences: boolean;\n  public sequenceColumnName: string | null;\n  public positionMarginState: string;\n  public positionMargin: number = 0;\n  public startPositionName: string | null;\n  public endPositionName: string | null;\n  public fixWidth: boolean;\n  public verticalAlignment: string | null;\n  public horizontalAlignment: string | null;\n  public fitArea: boolean;\n  public shrinkEmptyTail: boolean;\n  public skipEmptyPositions: boolean;\n  public positionHeight: string;\n\n  private positionNames: string[] = [];\n\n  private startPosition: number = -1;\n\n  private endPosition: number = -1;\n\n  /** For startPosition equals to endPosition Length is 1 */\n  private get Length(): number {\n    if (this.skipEmptyPositions) {\n      return this.positions.length;\n    }\n    return this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0;\n  }\n\n  private get positionWidthWithMargin() {\n    if ((this.positionMarginState === 'auto') && (this.unitsHandler?.getAlphabetIsMultichar() === true)) {\n      return this._positionWidth + this.positionMargin;\n    }\n    if (this.positionMarginState === 'enable') {\n      return this._positionWidth + this.positionMargin;\n    }\n\n    return this._positionWidth;\n  }\n\n  private viewSubs: Subscription[] = [];\n\n  constructor() {\n    super();\n\n    this.viewerId = WebLogo.viewerCount;\n    WebLogo.viewerCount += 1;\n\n    this.textBaseline = 'top';\n    this.unitsHandler = null;\n\n    this._positionWidth = this.positionWidth = this.float('positionWidth', 16/*,\n      {editor: 'slider', min: 4, max: 64, postfix: 'px'}*/);\n    this.minHeight = this.float('minHeight', 50/*,\n      {editor: 'slider', min: 25, max: 250, postfix: 'px'}*/);\n    this.maxHeight = this.float('maxHeight', 100/*,\n      {editor: 'slider', min: 25, max: 500, postfix: 'px'}*/);\n\n    this.considerNullSequences = this.bool('considerNullSequences', false);\n    this.sequenceColumnName = this.string('sequenceColumnName', null);\n\n    this.startPositionName = this.string('startPositionName', null);\n    this.endPositionName = this.string('endPositionName', null);\n\n    this.fixWidth = this.bool('fixWidth', false);\n\n    this.verticalAlignment = this.string('verticalAlignment', 'middle',\n      {choices: ['top', 'middle', 'bottom']});\n    this.horizontalAlignment = this.string('horizontalAlignment', 'center',\n      {choices: ['left', 'center', 'right']});\n    this.fitArea = this.bool('fitArea', true);\n    this.shrinkEmptyTail = this.bool('shrinkEmptyTail', true);\n    this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n    this.positionMarginState = this.string('positionMarginState', 'auto',\n      {choices: ['auto', 'enable', 'off']});\n    let defaultValueForPositionMargin = 0;\n    if (this.positionMarginState === 'auto') {\n        defaultValueForPositionMargin = 4;\n    }\n    this.positionMargin = this.int('positionMargin', defaultValueForPositionMargin, {min: 0, max: 16});\n    this.positionHeight = this.string('positionHeight', PositionHeight.full, {choices: [PositionHeight.full, PositionHeight.Entropy]});\n  }\n\n  private async init(): Promise<void> {\n    if (this.initialized) {\n      console.error('WebLogo second initialization!');\n      return;\n    }\n\n    this.initialized = true;\n    this.helpUrl = '/help/visualize/viewers/web-logo.md';\n\n    this.msgHost = ui.div('No message');\n    this.msgHost.style.display = 'none';\n\n    this.canvas = ui.canvas();\n    this.canvas.style.width = '100%';\n\n    this.host = ui.div([this.msgHost, this.canvas]);\n\n    // this.slider = ui.rangeSlider(0, 20, 2, 5);\n    // this.slider.root.style.width = '100%';\n    // this.slider.root.style.height = '12px';\n\n    const getMonomer = (p: DG.Point): [number, string | null, PositionMonomerInfo | null] => {\n      const jPos = Math.floor(p.x / this.positionWidthWithMargin);\n      const position = this.positions[jPos];\n\n      if (position === void 0)\n        return [jPos, null, null];\n\n      const monomer: string | undefined = Object.keys(position.freq)\n        .find((m) => position.freq[m].bounds.contains(p.x, p.y));\n      if (monomer === undefined)\n        return [jPos, null, null];\n\n      return [jPos, monomer, position.freq[monomer]];\n    };\n\n\n    // this.subs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mouseover').subscribe((e: MouseEvent) => {\n    // }));\n\n    this.viewSubs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mousemove').subscribe((e: MouseEvent) => {\n      if (!this.canvas)\n        return;\n\n      const args = e as MouseEvent;\n\n      const r: number = window.devicePixelRatio;\n      const cursorP: DG.Point = this.canvas.getCursorPosition(args, r);\n      const [jPos, monomer] = getMonomer(cursorP);\n\n      if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n        const rowCount = wu.count().take(this.dataFrame.rowCount).filter((iRow: number) => {\n          const seq = this.seqCol!.get(iRow);\n          const seqM = seq ? this.splitter!(seq)[this.startPosition + jPos] : null;\n          return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n        }).reduce<number>((count, iRow) => count + 1, 0);\n        ui.tooltip.show(ui.div([ui.div(`${monomer}`), ui.div(`${rowCount} rows`)]), args.x + 16, args.y + 16);\n      } else {\n        ui.tooltip.hide();\n      }\n    }));\n\n    this.viewSubs.push(rxjs.fromEvent<MouseEvent>(this.canvas, 'mousedown').subscribe((e: MouseEvent) => {\n      if (!this.canvas || e.button != 0)\n        return;\n\n      const args = e as MouseEvent;\n      const r: number = window.devicePixelRatio;\n      const [jPos, monomer] = getMonomer(this.canvas.getCursorPosition(args, r));\n\n      // prevents deselect all rows if we miss monomer bounds\n      if (this.dataFrame && this.seqCol && this.splitter && monomer) {\n        this.dataFrame.selection.init((iRow) => {\n          const seq = this.seqCol!.get(iRow);\n          const seqM = seq ? this.splitter!(seq)[this.startPosition + jPos] : null;\n          return ((seqM === monomer) || (seqM === '' && monomer === '-')) && this.dataFrame.filter.get(iRow);\n        });\n      }\n    }));\n\n    this.viewSubs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n\n    this.root.append(this.host);\n    // this.root.appendChild(this.slider.root);\n\n    this.render(true);\n  }\n\n  private rootOnSizeChanged(args: any) {\n    this.render(true);\n\n    // console.debug(`WebLogo.onRootSizeChanged() ` +\n    //   `root.width=${this.root.clientWidth}, root.height=${this.root.clientHeight}, ` +\n    //   `canvas.width=${this.canvas.width}, canvas.height=${this.canvas.height} .`);\n  }\n\n  /** Assigns {@link seqCol} and {@link cp} based on {@link sequenceColumnName} and calls {@link render}().\n   */\n  private updateSeqCol(): void {\n    if (this.dataFrame) {\n      this.seqCol = this.sequenceColumnName ? this.dataFrame.col(this.sequenceColumnName) : null;\n      if (this.seqCol == null) {\n        this.seqCol = WebLogo.pickUpSeqCol2(this.dataFrame);\n        this.sequenceColumnName = this.seqCol ? this.seqCol.name : null;\n      }\n      if (this.seqCol) {\n        const units: string = this.seqCol!.getTag(DG.TAGS.UNITS);\n        const separator: string = this.seqCol!.getTag(UnitsHandler.TAGS.separator);\n        this.splitter = WebLogo.getSplitter(units, separator);\n        this.unitsHandler = new UnitsHandler(this.seqCol);\n\n        this.updatePositions();\n        this.cp = WebLogo.pickUpPalette(this.seqCol);\n      } else {\n        this.splitter = null;\n        this.positionNames = [];\n        this.startPosition = -1;\n        this.endPosition = -1;\n        this.cp = null;\n      }\n    }\n    this.render();\n  }\n\n  private updatePositions(): void {\n    if (!this.seqCol)\n      return;\n\n    let categories: (string | null) [];\n    if (this.shrinkEmptyTail) {\n      const indices: Int32Array = this.dataFrame.filter.getSelectedIndexes();\n      categories = Array.from(new Set(\n        Array.from(Array(indices.length).keys()).map((i: number) => this.seqCol!.get(indices[i]))));\n    } else {\n      categories = this.seqCol.categories;\n    }\n    const maxLength = categories.length > 0 ? Math.max(...categories.map(\n      (s) => s !== null ? this.splitter!(s).length : 0)) : 0;\n\n    // Get position names from data column tag 'positionNames'\n    const positionNamesTxt = this.seqCol.getTag('positionNames');\n    // Fallback if 'positionNames' tag is not provided\n    this.positionNames = positionNamesTxt ? positionNamesTxt.split(', ').map((n) => n.trim()) :\n      [...Array(maxLength).keys()].map((jPos) => `${jPos + 1}`);\n\n    this.startPosition = (this.startPositionName && this.positionNames &&\n      this.positionNames.includes(this.startPositionName)) ?\n      this.positionNames.indexOf(this.startPositionName) : 0;\n    this.endPosition = (this.endPositionName && this.positionNames &&\n      this.positionNames.includes(this.endPositionName)) ?\n      this.positionNames.indexOf(this.endPositionName) : (maxLength - 1);\n  }\n\n  public override onPropertyChanged(property: DG.Property): void {\n    // console.debug(`WebLogo.onPropertyChanged( ${property.name} = '' })`);\n    super.onPropertyChanged(property);\n\n    switch (property.name) {\n    case 'considerNullSequences':\n      this.render(true);\n      break;\n    case 'sequenceColumnName':\n      this.updateSeqCol();\n      break;\n    case 'startPositionName':\n      this.updateSeqCol();\n      break;\n    case 'endPositionName':\n      this.updateSeqCol();\n      break;\n    case 'positionWidth':\n      this._positionWidth = this.positionWidth;\n      this.render(true);\n      break;\n    case 'minHeight':\n      this.render(true);\n      break;\n    case 'maxHeight':\n      this.render(true);\n      break;\n    case 'fixWidth':\n      this.render(true);\n      break;\n    case 'verticalAlignment':\n      this.render(true);\n      break;\n    case 'horizontalAlignment':\n      this.render(true);\n      break;\n    case 'fitArea':\n      this.render(true);\n      break;\n    case 'shrinkEmptyTail':\n      this.updatePositions();\n      this.render(true);\n      break;\n    case 'skipEmptyPositions':\n      this.updatePositions();\n      this.render(true);\n      break;\n    case 'positionMargin':\n      this.render(true);\n      break;\n    case 'positionMarginState':\n      this.render(true);\n      break;\n    case 'positionHeight':\n      this.render(true);\n      break;\n    }\n  }\n\n  public override async onTableAttached() {\n    const dataFrameTxt: string = this.dataFrame ? 'data' : 'null';\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n    super.onTableAttached();\n\n    this.updateSeqCol();\n\n    if (this.dataFrame !== void 0) {\n      this.subs.push(this.dataFrame.selection.onChanged.subscribe((_) => this.render()));\n      this.subs.push(this.dataFrame.filter.onChanged.subscribe((_) => {\n        this.updatePositions();\n        this.render();\n      }));\n    }\n\n    await this.init();\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n  }\n\n  public override async detach() {\n    const dataFrameTxt = `${this.dataFrame ? 'data' : 'null'}`;\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached( dataFrame = ${dataFrameTxt} ) start`);\n    super.detach();\n\n    this.viewSubs.forEach((sub) => sub.unsubscribe());\n    this.host!.remove();\n    this.canvas = undefined;\n    this.msgHost = undefined;\n    this.host = undefined;\n\n    this.initialized = false;\n    console.debug(`bio: WebLogo<${this.viewerId}>.onTableAttached() end`);\n  }\n\n  protected _nullSequence(fillerResidue = 'X'): string {\n    if (this.considerNullSequences)\n      return new Array(this.Length).fill(fillerResidue).join('');\n\n    return '';\n  }\n\n  protected removeWhere(array: Array<any>, predicate: (T: any) => boolean): Array<any> {\n    let length = array.length;\n    let updateIterator = 0;\n    for (let deleteIterator = 0; deleteIterator < length; deleteIterator++) {\n      if (!predicate(array[deleteIterator])) {\n        array[updateIterator] = array[deleteIterator];\n        updateIterator++;\n      }\n    }\n    array.length = updateIterator;\n    return array;\n  }\n\n\n  protected _removeEmptyPositions() {\n    if (this.skipEmptyPositions) {\n      this.removeWhere(this.positions, item => item?.freq['-']?.count === item.rowCount);\n    }\n  }\n\n  protected _calculate(r: number) {\n    if (!this.canvas || !this.host || !this.seqCol || !this.dataFrame)\n      return;\n    this.unitsHandler = new UnitsHandler(this.seqCol);\n\n\n    this.calcSize();\n\n    this.positions = new Array(this.startPosition <= this.endPosition ? this.endPosition - this.startPosition + 1 : 0);\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const posName: string = this.positionNames[this.startPosition + jPos];\n      this.positions[jPos] = new PositionInfo(posName);\n    }\n\n    // 2022-05-05 askalkin instructed to show WebLogo based on filter (not selection)\n    const indices = this.dataFrame.filter.getSelectedIndexes();\n    // const indices = this.dataFrame.selection.trueCount > 0 ? this.dataFrame.selection.getSelectedIndexes() :\n    //   this.dataFrame.filter.getSelectedIndexes();\n\n    this.rowsMasked = indices.length;\n    this.rowsNull = 0;\n\n    for (const i of indices) {\n      let s: string = <string>(this.seqCol.get(i));\n\n      if (!s) {\n        s = this._nullSequence();\n        ++this.rowsNull;\n      }\n\n      const seqM: string[] = this.splitter!(s);\n      for (let jPos = 0; jPos < this.Length; jPos++) {\n        const pmInfo = this.positions[jPos].freq;\n        const m: string = seqM[this.startPosition + jPos] || '-';\n        if (!(m in pmInfo))\n          pmInfo[m] = new PositionMonomerInfo();\n        pmInfo[m].count++;\n      }\n    }\n\n    //#region Polish freq counts\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      // delete this.positions[jPos].freq['-'];\n\n      this.positions[jPos].rowCount = 0;\n      for (const m in this.positions[jPos].freq)\n        this.positions[jPos].rowCount += this.positions[jPos].freq[m].count;\n      if (this.positionHeight == PositionHeight.Entropy) {\n        this.positions[jPos].sumForHeightCalc = 0;\n        for (const m in this.positions[jPos].freq) {\n          const pn = this.positions[jPos].freq[m].count / this.positions[jPos].rowCount;\n          this.positions[jPos].sumForHeightCalc += -pn * Math.log2(pn);\n        }\n      }\n    }\n    //#endregion\n    this._removeEmptyPositions();\n\n    const absoluteMaxHeight = this.canvas.height - this.axisHeight * r;\n    // console.debug(`WebLogo<${this.viewerId}>._calculate() maxHeight=${maxHeight}.`);\n\n    //#region Calculate screen\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const freq: { [c: string]: PositionMonomerInfo } = this.positions[jPos].freq;\n      const rowCount = this.positions[jPos].rowCount;\n      const alphabetSize = this.getAlphabetSize();\n      if ((this.positionHeight == PositionHeight.Entropy) && (alphabetSize == null)) {\n        grok.shell.error('WebLogo: alphabet is undefined.');\n      }\n\n      const maxHeight = (this.positionHeight == PositionHeight.Entropy) ? (absoluteMaxHeight * (Math.log2(alphabetSize) - (this.positions[jPos].sumForHeightCalc)) / Math.log2(alphabetSize)) : absoluteMaxHeight;\n\n      let y: number = this.axisHeight * r + (absoluteMaxHeight - maxHeight - 1);\n\n      const entries = Object.entries(freq).sort((a, b) => {\n        if (a[0] !== '-' && b[0] !== '-')\n          return b[1].count - a[1].count;\n        else if (a[0] === '-' && b[0] === '-')\n          return 0;\n        else if (a[0] === '-')\n          return -1;\n        else /* (b[0] === '-') */\n          return +1;\n      });\n      for (const entry of entries) {\n        const pmInfo: PositionMonomerInfo = entry[1];\n        // const m: string = entry[0];\n        const h: number = maxHeight * pmInfo.count / rowCount;\n\n        pmInfo.bounds = new DG.Rect(jPos * this.positionWidthWithMargin, y, this._positionWidth, h);\n        y += h;\n      }\n    }\n    //#endregion\n  }\n\n  // reflect changes made to filter/selection\n  render(recalc = true) {\n    if (this.msgHost) {\n      if (this.seqCol && !this.cp) {\n        this.msgHost!.innerText = `Unknown palette (column semType: '${this.seqCol.semType}').`;\n        this.msgHost!.style.display = '';\n      } else {\n        this.msgHost!.style.display = 'none';\n      }\n    }\n\n    if (!this.canvas || !this.seqCol || !this.dataFrame || !this.cp)\n      return;\n\n    const g = this.canvas.getContext('2d');\n    if (!g) return;\n\n    const r = window.devicePixelRatio;\n\n    if (recalc)\n      this._calculate(r);\n\n    g.resetTransform();\n    g.fillStyle = 'white';\n    g.fillRect(0, 0, this.canvas.width, this.canvas.height);\n    g.textBaseline = this.textBaseline;\n\n    // Prevents division by zero on Length = 0\n    if (this.startPosition === -1 || this.endPosition === -1)\n      return;\n\n    //#region Plot positionNames\n    const positionFontSize = 10 * r;\n    g.resetTransform();\n    g.fillStyle = 'black';\n    g.textAlign = 'center';\n    g.font = `${positionFontSize.toFixed(1)}px Roboto, Roboto Local, sans-serif`;\n    const posNameMaxWidth = Math.max(...this.positions.map((pos) => g.measureText(pos.name).width));\n    const hScale = posNameMaxWidth < (this._positionWidth - 2) ? 1 : (this._positionWidth - 2) / posNameMaxWidth;\n\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      const pos: PositionInfo = this.positions[jPos];\n      g.resetTransform();\n      g.setTransform(\n        hScale, 0, 0, 1,\n        jPos * this.positionWidthWithMargin + this._positionWidth / 2, 0);\n      g.fillText(pos.name, 0, 0);\n    }\n    //#endregion Plot positionNames\n\n    for (let jPos = 0; jPos < this.Length; jPos++) {\n      for (const [monomer, pmInfo] of Object.entries(this.positions[jPos].freq)) {\n        if (monomer !== '-') {\n          const monomerTxt = WebLogo.monomerToText(monomer);\n          const b = pmInfo.bounds;\n\n          const fontStyle = '16px Roboto, Roboto Local, sans-serif';\n          // Hacks to scale uppercase characters to target rectangle\n          const uppercaseLetterAscent = 0.25;\n          const uppercaseLetterHeight = 12.2;\n\n          g.resetTransform();\n          g.strokeStyle = 'lightgray';\n          g.lineWidth = 1;\n          g.rect(b.left, b.top, b.width, b.height);\n          g.fillStyle = this.cp.get(monomer) ?? this.cp.get('other');\n          g.textAlign = 'left';\n          g.font = fontStyle;\n          //g.fillRect(b.left, b.top, b.width, b.height);\n          const mTm: TextMetrics = g.measureText(monomerTxt);\n\n          // if (mM.actualBoundingBoxAscent != 0)\n          //   console.debug(`m: ${m}, mM.actualBoundingBoxAscent: ${mM.actualBoundingBoxAscent}`);\n\n          g.setTransform(\n            b.width / mTm.width, 0, 0, b.height / uppercaseLetterHeight,\n            b.left, b.top);\n          g.fillText(monomerTxt, 0, -uppercaseLetterAscent);\n        }\n      }\n    }\n  }\n\n  private calcSize() {\n    if (!this.canvas || !this.host)\n      return;\n\n    const r: number = window.devicePixelRatio;\n\n    let width: number = this.Length * this.positionWidthWithMargin / r;\n    let height = Math.min(this.maxHeight, Math.max(this.minHeight, this.root.clientHeight));\n\n    if (this.fitArea) {\n      const xScale: number = this.root.clientHeight / height;\n      const yScale: number = this.root.clientWidth / width;\n      const scale = Math.max(1, Math.min(xScale, yScale));\n      width = width * scale;\n      height = height * scale;\n      this._positionWidth = this.positionWidth * scale;\n    }\n\n    this.canvas.width = width * r;\n    this.canvas.style.width = `${width}px`;\n\n    // const canvasHeight: number = width > this.root.clientWidth ? height - 8 : height;\n    this.host.style.setProperty('height', `${height}px`);\n    const canvasHeight: number = this.host.clientHeight;\n    this.canvas.height = canvasHeight * r;\n    this.canvas.style.setProperty('height', `${canvasHeight}px`);\n\n    // Adjust host and root width\n    if (this.fixWidth) {\n      // full width for canvas host and root\n      this.root.style.width = this.host.style.width = `${width}px`;\n      this.root.style.height = `${height}px`;\n      this.host.style.setProperty('overflow', 'hidden', 'important');\n    } else {\n      // allow scroll canvas in root\n      this.root.style.width = this.host.style.width = '100%';\n      this.host.style.overflowX = 'auto!important';\n      this.host.style.setProperty('overflow', null);\n\n      this.host.style.setProperty('text-align', this.horizontalAlignment);\n\n      // vertical alignment\n      let hostTopMargin = 0;\n      switch (this.verticalAlignment) {\n      case 'top':\n        hostTopMargin = 0;\n        break;\n      case 'middle':\n        hostTopMargin = Math.max(0, (this.root.clientHeight - height) / 2);\n        break;\n      case 'bottom':\n        hostTopMargin = Math.max(0, this.root.clientHeight - height);\n        break;\n      }\n      this.host.style.setProperty('margin-top', `${hostTopMargin}px`, 'important');\n\n      if (this.root.clientHeight < height) {\n        this.host.style.setProperty('height', `${this.root.clientHeight}px`);\n        this.host.style.setProperty('overflow-y', null);\n      } else {\n        this.host.style.setProperty('overflow-y', 'hidden', 'important');\n      }\n    }\n\n    // console.debug(\n    //   `this.root.style.height = ${this.root.style.height}\\n` +\n    //   `this.root.clientHeight = ${this.root.clientHeight}\\n` +\n    //   `this.host.style.height = ${this.host.style.height}\\n` +\n    //   `this.host.clientHeight = ${this.host.clientHeight}\\n` +\n    //   '\\n' +\n    //   `this.canvas.height       = ${this.canvas.height}\\n` +\n    //   `this.canvas.style.height = ${this.canvas.style.height}`);\n  }\n\n  /**\n   * @param {DG.Column} seqCol Column to look for a palette\n   * @param {number}  minLength minimum length of sequence to detect palette (empty strings are allowed)\n   * @return {SeqPalette} Palette corresponding to the alphabet of the sequences in the column\n   */\n  public static pickUpPalette(seqCol: DG.Column, minLength: number = 5): SeqPalette {\n    let res: SeqPalette | null = null;\n    switch (seqCol.semType) {\n    case Aminoacids.SemTypeMultipleAlignment:\n      res = AminoacidsPalettes.GrokGroups;\n      break;\n    case Nucleotides.SemTypeMultipleAlignment:\n      res = NucleotidesPalettes.Chromatogram;\n      break;\n    }\n    const stats: SeqColStats = WebLogo.getStats(seqCol, minLength, WebLogo.splitterAsFasta);\n\n    const alphabetCandidates: [Set<string>, SeqPalette][] = [\n      [new Set(Object.keys(Nucleotides.Names)), NucleotidesPalettes.Chromatogram],\n      [new Set(Object.keys(Aminoacids.Names)), AminoacidsPalettes.GrokGroups],\n    ];\n    // Calculate likelihoods for alphabet_candidates\n    const alphabetCandidatesSim: number[] = alphabetCandidates\n      .map((c) => WebLogo.getAlphabetSimilarity(stats.freq, c[0]));\n    const maxCos = Math.max(...alphabetCandidatesSim);\n    if (maxCos > 0.55)\n      res = alphabetCandidates[alphabetCandidatesSim.indexOf(maxCos)][1];\n    else\n      res = UnknownSeqPalettes.Color;\n\n    return res;\n  }\n\n  public getAlphabetSize(): number {\n    return this.unitsHandler?.getAlphabetSize() ?? 0;\n  }\n\n  /** Stats of sequences with specified splitter func, returns { freq, sameLength }.\n   * @param {DG.Column} seqCol\n   * @param {number} minLength\n   * @param {SplitterFunc} splitter\n   * @return { SeqColStats }, sameLength: boolean } stats of column sequences\n   */\n  static getStats(seqCol: DG.Column, minLength: number, splitter: SplitterFunc): SeqColStats {\n    const freq: { [m: string]: number } = {};\n    let sameLength = true;\n    let firstLength = null;\n\n    for (const seq of seqCol.categories) {\n      const mSeq = splitter(seq);\n\n      if (firstLength == null)\n        firstLength = mSeq.length;\n      else if (mSeq.length !== firstLength)\n        sameLength = false;\n\n      if (mSeq.length > minLength) {\n        for (const m of mSeq) {\n          if (!(m in freq))\n            freq[m] = 0;\n          freq[m] += 1;\n        }\n      }\n    }\n    return {freq: freq, sameLength: sameLength};\n  }\n\n  public static getAlphabetSimilarity(freq: MonomerFreqs, alphabet: Set<string>, gapSymbol: string = '-'): number {\n    const keys = new Set<string>([...new Set(Object.keys(freq)), ...alphabet]);\n    keys.delete(gapSymbol);\n\n    const freqA: number[] = [];\n    const alphabetA: number[] = [];\n    for (const m of keys) {\n      freqA.push(m in freq ? freq[m] : 0);\n      alphabetA.push(alphabet.has(m) ? 1 : 0);\n    }\n    /* There were a few ideas: chi-squared, pearson correlation (variance?), scalar product */\n    const freqV: Vector = new Vector(freqA);\n    const alphabetV: Vector = new Vector(alphabetA);\n    return vectorDotProduct(freqV, alphabetV) / (vectorLength(freqV) * vectorLength(alphabetV));\n  }\n\n  // /** First try to find column with semType 'alignedSequence'.\n  //  * Next look for column with data alphabet corresponding to any of the known palettes.\n  //  * @param {DG.DataFrame} dataFrame\n  //  * @return {DG.Column} The column we were looking for or null\n  //  */\n  // private pickUpSeqCol(dataFrame: DG.DataFrame): DG.Column | null {\n  //   let res: DG.Column | null = dataFrame.columns.bySemType('alignedSequence');\n  //   if (res == null) {\n  //     for (const col of dataFrame.columns) {\n  //       const cp = WebLogo.pickUpPalette(col as DG.Column, 5);\n  //       if (cp !== null && !(cp instanceof UnknownSeqPalette)) {\n  //         res = col;\n  //         break;\n  //       }\n  //     }\n  //   }\n  //   return res;\n  // }\n\n  public static pickUpSeqCol2(df: DG.DataFrame): DG.Column | null {\n    const semTypeColList = df.columns.bySemTypeAll(DG.SEMTYPE.MACROMOLECULE);\n    let resCol: DG.Column | null = semTypeColList.find((col) => {\n      const units = col.getTag(DG.TAGS.UNITS);\n      return units ? units.indexOf('MSA') !== -1 : false;\n    }) ?? null;\n    if (!resCol && semTypeColList.length > 0)\n      resCol = semTypeColList[0];\n    return resCol;\n  }\n\n  private static monomerRe = /\\[(\\w+)\\]|(\\w)|(-)/g;\n\n  /** Split sequence for single character monomers, square brackets multichar monomer names or gap symbol.\n   * @param {any} seq object with sequence\n   * @return {string[]} array of monomers\n   */\n  public static splitterAsFasta(seq: any): string[] {\n    const res: string[] = wu<RegExpMatchArray>(seq.toString().matchAll(WebLogo.monomerRe))\n      .map((ma: RegExpMatchArray) => {\n        let mRes: string;\n        const m: string = ma[0];\n        if (m.length > 1) {\n          if (m in WebLogo.aaSynonyms) {\n            mRes = WebLogo.aaSynonyms[m];\n          } else {\n            mRes = '';\n            console.debug(`Long monomer '${m}' has not a short synonym.`);\n          }\n        } else {\n          mRes = m;\n        }\n        return mRes;\n      }).toArray();\n\n    return res;\n  }\n\n  private static helmRe = /(PEPTIDE1|DNA1|RNA1)\\{([^}]+)}/g;\n  private static helmPp1Re = /\\[([^\\[\\]]+)]/g;\n\n  /** Splits Helm string to monomers, but does not replace monomer names to other notation (e.g. for RNA).\n   * Only for linear polymers, does not split RNA for ribose and phosphate monomers.\n   * @param {string} seq Source string of HELM notation\n   * @return {string[]}\n   */\n  public static splitterAsHelm(seq: any): string[] {\n    WebLogo.helmRe.lastIndex = 0;\n    const ea: RegExpExecArray | null = WebLogo.helmRe.exec(seq.toString());\n    const inSeq: string | null = ea ? ea[2] : null;\n\n    const mmPostProcess = (mm: string): string => {\n      WebLogo.helmPp1Re.lastIndex = 0;\n      const pp1M = WebLogo.helmPp1Re.exec(mm);\n      if (pp1M && pp1M.length >= 2)\n        return pp1M[1];\n      else\n        return mm;\n    };\n\n    const mmList: string[] = inSeq ? inSeq.split('.') : [];\n    const mmListRes: string[] = mmList.map(mmPostProcess);\n    return mmListRes;\n  }\n\n  /** Gets method to split sequence by separator\n   * @param {string} separator\n   * @param limit\n   * @return {SplitterFunc}\n   */\n  public static getSplitterWithSeparator(separator: string, limit: number | undefined = undefined): SplitterFunc {\n    return (seq: string) => {\n      return seq.split(separator, limit);\n    };\n  }\n\n  /** Get splitter method to split sequences to monomers\n   * @param {string} units\n   * @param {string} separator\n   * @param limit\n   * @return {SplitterFunc}\n   */\n  public static getSplitter(units: string, separator: string, limit: number | undefined = undefined): SplitterFunc {\n    if (units.toLowerCase().startsWith(NOTATION.FASTA))\n      return WebLogo.splitterAsFasta;\n    else if (units.toLowerCase().startsWith(NOTATION.SEPARATOR))\n      return WebLogo.getSplitterWithSeparator(separator, limit);\n    else if (units.toLowerCase().startsWith(NOTATION.HELM))\n      return WebLogo.splitterAsHelm;\n    else\n      throw new Error(`Unexpected units ${units} .`);\n\n    // TODO: Splitter for HELM\n  }\n\n  public static getSplitterForColumn(col: DG.Column): SplitterFunc {\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE)\n      throw new Error(`Get splitter for semType \"${DG.SEMTYPE.MACROMOLECULE}\" only.`);\n\n    const units = col.getTag(DG.TAGS.UNITS);\n    const separator = col.getTag(UnitsHandler.TAGS.separator);\n    return WebLogo.getSplitter(units, separator);\n  }\n\n  /** Only some of the synonyms. These were obtained from the clustered oligopeptide dataset. */\n  private static aaSynonyms: { [name: string]: string } = {\n    '[MeNle]': 'L', // Nle - norleucine\n    '[MeA]': 'A',\n    '[MeG]': 'G',\n    '[MeF]': 'F',\n  };\n\n  private static longMonomerPartRe = /(\\w+)/g;\n\n  /* Shortens monomer representation text for monomers with long names.\n  **/\n  public static monomerToText(src: any): string {\n    const srcTxt: string = src.toString();\n    if (srcTxt.length <= 5) {\n      return srcTxt;\n    } else {\n      const parts: string[] = wu<RegExpMatchArray>(src.toString().matchAll(WebLogo.longMonomerPartRe))\n        .map((ma: RegExpMatchArray) => {\n          const mRes: string = ma[0];\n          return mRes;\n        }).toArray();\n\n      if (parts.length == 0) {\n        return ' ';\n      } else {\n        const part0: string = parts[0];\n        const resTxt: string = part0.length < 6 ? `${part0}…` : `${part0.substring(0, 5)}…`;\n        return resTxt;\n      }\n    }\n  }\n\n  public static monomerToShort(amino: string, maxLengthOfMonomer: number): string {\n    return amino.length <= maxLengthOfMonomer ? amino : amino.substring(0, maxLengthOfMonomer) + '…';\n  }\n}\n"]}","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport { normalize } from '@datagrok-libraries/utils/src/vector-operations';\nimport { createDimensinalityReducingWorker } from './workers/dimensionality-reducing-worker-creator';\nexport function reduceDimensinalityWithNormalization(dataCol, methodName, similarityMetric, options) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const dimensionalityReduceRes = yield createDimensinalityReducingWorker({ data: dataCol, metric: similarityMetric }, methodName, options);\n        dimensionalityReduceRes.embedding = dimensionalityReduceRes.embedding.map(it => normalize(it));\n        return dimensionalityReduceRes;\n    });\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxdWVuY2Utc3BhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzZXF1ZW5jZS1zcGFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFFQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0saURBQWlELENBQUM7QUFHNUUsT0FBTyxFQUFFLGlDQUFpQyxFQUErQixNQUFNLGtEQUFrRCxDQUFDO0FBR2xJLE1BQU0sVUFBZ0Isb0NBQW9DLENBQ3RELE9BQXFDLEVBQ3JDLFVBQWtCLEVBQ2xCLGdCQUE2RCxFQUM3RCxPQUFhOztRQUViLE1BQU0sdUJBQXVCLEdBQ3pCLE1BQU0saUNBQWlDLENBQ25DLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsZ0JBQWdCLEVBQWdCLEVBQ3pELFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUU3Qix1QkFBdUIsQ0FBQyxTQUFTLEdBQUcsdUJBQXVCLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQy9GLE9BQU8sdUJBQXVCLENBQUM7SUFDbkMsQ0FBQztDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJpdEFycmF5IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL2JpdC1hcnJheSc7XG5pbXBvcnQgeyBNYXRyaXgsIFZlY3RvciB9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3R5cGUtZGVjbGFyYXRpb25zJztcbmltcG9ydCB7IG5vcm1hbGl6ZSB9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3ZlY3Rvci1vcGVyYXRpb25zJztcbmltcG9ydCAqIGFzIERHIGZyb20gJ2RhdGFncm9rLWFwaS9kZyc7XG5pbXBvcnQgeyBCaXRBcnJheU1ldHJpY3MsIFN0cmluZ01ldHJpY3MsIFZhbGlkVHlwZXMsIFZlY3Rvck1ldHJpY3MgfSBmcm9tICcuL3R5cGVkLW1ldHJpY3MnO1xuaW1wb3J0IHsgY3JlYXRlRGltZW5zaW5hbGl0eVJlZHVjaW5nV29ya2VyLCBJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQgfSBmcm9tICcuL3dvcmtlcnMvZGltZW5zaW9uYWxpdHktcmVkdWNpbmctd29ya2VyLWNyZWF0b3InO1xuXG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiByZWR1Y2VEaW1lbnNpbmFsaXR5V2l0aE5vcm1hbGl6YXRpb24oXG4gICAgZGF0YUNvbDogQml0QXJyYXlbXXxWZWN0b3JbXXxzdHJpbmdbXSwgXG4gICAgbWV0aG9kTmFtZTogc3RyaW5nLCBcbiAgICBzaW1pbGFyaXR5TWV0cmljOiBCaXRBcnJheU1ldHJpY3N8VmVjdG9yTWV0cmljc3xTdHJpbmdNZXRyaWNzLFxuICAgIG9wdGlvbnM/OiBhbnkpOiBQcm9taXNlPElSZWR1Y2VEaW1lbnNpb25hbGl0eVJlc3VsdD4ge1xuXG4gICAgY29uc3QgZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXM6IElSZWR1Y2VEaW1lbnNpb25hbGl0eVJlc3VsdCA9XG4gICAgICAgIGF3YWl0IGNyZWF0ZURpbWVuc2luYWxpdHlSZWR1Y2luZ1dvcmtlcihcbiAgICAgICAgICAgIHsgZGF0YTogZGF0YUNvbCwgbWV0cmljOiBzaW1pbGFyaXR5TWV0cmljIH0gYXMgVmFsaWRUeXBlcyxcbiAgICAgICAgICAgIG1ldGhvZE5hbWUsIG9wdGlvbnMpO1xuXG4gICAgZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXMuZW1iZWRkaW5nID0gZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXMuZW1iZWRkaW5nLm1hcChpdCA9PiBub3JtYWxpemUoaXQpKTtcbiAgICByZXR1cm4gZGltZW5zaW9uYWxpdHlSZWR1Y2VSZXM7XG59Il19","import * as fl from 'fastest-levenshtein';\nimport { jaroWinkler } from 'jaro-winkler-typescript';\nimport { distanceMetrics } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { calculateEuclideanDistance } from '@datagrok-libraries/utils/src/vector-operations';\nexport const AvailableMetrics = {\n    'Vector': {\n        'EuclideanDistance': calculateEuclideanDistance,\n    },\n    'String': {\n        'Levenshtein': fl.distance,\n        'Jaro-Winkler': jaroWinkler,\n    },\n    'BitArray': {\n        'Tanimoto': distanceMetrics['Tanimoto'],\n        'Dice': distanceMetrics['Dice'],\n        'Asymmetric': distanceMetrics['Asymmetric'],\n        'Braun-Blanquet': distanceMetrics['Braun-Blanquet'],\n        'Cosine': distanceMetrics['Cosine'],\n        'Kulczynski': distanceMetrics['Kulczynski'],\n        'Mc-Connaughey': distanceMetrics['Mc-Connaughey'],\n        'Rogot-Goldberg': distanceMetrics['Rogot-Goldberg'],\n        'Russel': distanceMetrics['Russel'],\n        'Sokal': distanceMetrics['Sokal'],\n    },\n};\nexport const MetricToDataType = Object.keys(AvailableMetrics)\n    .reduce((ret, key) => {\n    for (const val of Object.keys(AvailableMetrics[key])) {\n        ret[val] = key;\n    }\n    return ret;\n}, {});\nexport function isStringMetric(name) {\n    return MetricToDataType[name] == 'String';\n}\nexport function isBitArrayMetric(name) {\n    return MetricToDataType[name] == 'BitArray';\n}\nexport function isVectorMetric(name) {\n    return MetricToDataType[name] == 'Vector';\n}\n/** Unified class implementing different string measures. */\nexport class Measure {\n    /**\n     * Creates an instance of Measure with .\n     * @param {string} method Method to calculate distance between strings.\n     * @memberof Measurer\n     */\n    constructor(method) {\n        this.method = method;\n        this.dataType = MetricToDataType[method];\n    }\n    /**\n     * Returns custom string distance function specified.\n     * @return {DistanceMetric} Callback of the measure chosen.\n     * @memberof Measurer\n     */\n    getMeasure() {\n        const dict = AvailableMetrics;\n        return dict[this.dataType][this.method];\n    }\n    /**\n     * Returns custom string distance by the given data type.\n     * @param {AvailableDataTypes} dataType Metric's data type\n     * @return {string[]} Metric names which expects the given data type\n     * @memberof Measurer\n     */\n    static getMetricByDataType(dataType) {\n        return Object.keys(AvailableMetrics[dataType]);\n    }\n    /** Returns metric names available.\n     * @memberof Measurer\n    */\n    static get availableMeasures() {\n        return Object.keys(AvailableMetrics);\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWQtbWV0cmljcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInR5cGVkLW1ldHJpY3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUMxQyxPQUFPLEVBQUMsV0FBVyxFQUFDLE1BQU0seUJBQXlCLENBQUM7QUFFcEQsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLGtEQUFrRCxDQUFDO0FBQ2pGLE9BQU8sRUFBQywwQkFBMEIsRUFBQyxNQUFNLGlEQUFpRCxDQUFDO0FBSTNGLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHO0lBQzlCLFFBQVEsRUFBRTtRQUNSLG1CQUFtQixFQUFFLDBCQUEwQjtLQUNoRDtJQUNELFFBQVEsRUFBRTtRQUNSLGFBQWEsRUFBRSxFQUFFLENBQUMsUUFBUTtRQUMxQixjQUFjLEVBQUUsV0FBVztLQUM1QjtJQUNELFVBQVUsRUFBRTtRQUNWLFVBQVUsRUFBRSxlQUFlLENBQUMsVUFBVSxDQUFDO1FBQ3ZDLE1BQU0sRUFBRSxlQUFlLENBQUMsTUFBTSxDQUFDO1FBQy9CLFlBQVksRUFBRSxlQUFlLENBQUMsWUFBWSxDQUFDO1FBQzNDLGdCQUFnQixFQUFFLGVBQWUsQ0FBQyxnQkFBZ0IsQ0FBQztRQUNuRCxRQUFRLEVBQUUsZUFBZSxDQUFDLFFBQVEsQ0FBQztRQUNuQyxZQUFZLEVBQUUsZUFBZSxDQUFDLFlBQVksQ0FBQztRQUMzQyxlQUFlLEVBQUUsZUFBZSxDQUFDLGVBQWUsQ0FBQztRQUNqRCxnQkFBZ0IsRUFBRSxlQUFlLENBQUMsZ0JBQWdCLENBQUM7UUFDbkQsUUFBUSxFQUFFLGVBQWUsQ0FBQyxRQUFRLENBQUM7UUFDbkMsT0FBTyxFQUFFLGVBQWUsQ0FBQyxPQUFPLENBQUM7S0FDbEM7Q0FDRixDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sZ0JBQWdCLEdBQXFCLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUM7S0FDNUUsTUFBTSxDQUFDLENBQUMsR0FBcUIsRUFBRSxHQUFHLEVBQUUsRUFBRTtJQUNyQyxLQUFLLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsR0FBeUIsQ0FBQyxDQUFDLEVBQUU7UUFDMUUsR0FBRyxDQUFDLEdBQXlCLENBQUMsR0FBRyxHQUFHLENBQUM7S0FDdEM7SUFDRCxPQUFPLEdBQUcsQ0FBQztBQUNiLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztBQVdULE1BQU0sVUFBVSxjQUFjLENBQUMsSUFBa0I7SUFDL0MsT0FBTyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxRQUFRLENBQUM7QUFDNUMsQ0FBQztBQUVELE1BQU0sVUFBVSxnQkFBZ0IsQ0FBQyxJQUFrQjtJQUNqRCxPQUFPLGdCQUFnQixDQUFDLElBQUksQ0FBQyxJQUFJLFVBQVUsQ0FBQztBQUM5QyxDQUFDO0FBRUQsTUFBTSxVQUFVLGNBQWMsQ0FBQyxJQUFrQjtJQUMvQyxPQUFPLGdCQUFnQixDQUFDLElBQUksQ0FBQyxJQUFJLFFBQVEsQ0FBQztBQUM1QyxDQUFDO0FBRUQsNERBQTREO0FBQzVELE1BQU0sT0FBTyxPQUFPO0lBSWxCOzs7O09BSUc7SUFDSCxZQUFZLE1BQW9CO1FBQzlCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO1FBQ3JCLElBQUksQ0FBQyxRQUFRLEdBQUcsZ0JBQWdCLENBQUMsTUFBTSxDQUF1QixDQUFDO0lBQ2pFLENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksVUFBVTtRQUNmLE1BQU0sSUFBSSxHQUFzRCxnQkFBZ0IsQ0FBQztRQUNqRixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzFDLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxRQUE0QjtRQUM1RCxPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztJQUNqRCxDQUFDO0lBRUQ7O01BRUU7SUFDRixNQUFNLEtBQUssaUJBQWlCO1FBQzFCLE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGZsIGZyb20gJ2Zhc3Rlc3QtbGV2ZW5zaHRlaW4nO1xuaW1wb3J0IHtqYXJvV2lua2xlcn0gZnJvbSAnamFyby13aW5rbGVyLXR5cGVzY3JpcHQnO1xuaW1wb3J0IHtEaXN0YW5jZU1ldHJpY30gZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvdHlwZS1kZWNsYXJhdGlvbnMnO1xuaW1wb3J0IHtkaXN0YW5jZU1ldHJpY3N9IGZyb20gJ0BkYXRhZ3Jvay1saWJyYXJpZXMvdXRpbHMvc3JjL3NpbWlsYXJpdHktbWV0cmljcyc7XG5pbXBvcnQge2NhbGN1bGF0ZUV1Y2xpZGVhbkRpc3RhbmNlfSBmcm9tICdAZGF0YWdyb2stbGlicmFyaWVzL3V0aWxzL3NyYy92ZWN0b3Itb3BlcmF0aW9ucyc7XG5pbXBvcnQgQml0QXJyYXkgZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvYml0LWFycmF5JztcbmltcG9ydCB7VmVjdG9yLCBTdHJpbmdEaWN0aW9uYXJ5fSBmcm9tICdAZGF0YWdyb2stbGlicmFyaWVzL3V0aWxzL3NyYy90eXBlLWRlY2xhcmF0aW9ucyc7XG5cbmV4cG9ydCBjb25zdCBBdmFpbGFibGVNZXRyaWNzID0ge1xuICAnVmVjdG9yJzoge1xuICAgICdFdWNsaWRlYW5EaXN0YW5jZSc6IGNhbGN1bGF0ZUV1Y2xpZGVhbkRpc3RhbmNlLFxuICB9LFxuICAnU3RyaW5nJzoge1xuICAgICdMZXZlbnNodGVpbic6IGZsLmRpc3RhbmNlLFxuICAgICdKYXJvLVdpbmtsZXInOiBqYXJvV2lua2xlcixcbiAgfSxcbiAgJ0JpdEFycmF5Jzoge1xuICAgICdUYW5pbW90byc6IGRpc3RhbmNlTWV0cmljc1snVGFuaW1vdG8nXSxcbiAgICAnRGljZSc6IGRpc3RhbmNlTWV0cmljc1snRGljZSddLFxuICAgICdBc3ltbWV0cmljJzogZGlzdGFuY2VNZXRyaWNzWydBc3ltbWV0cmljJ10sXG4gICAgJ0JyYXVuLUJsYW5xdWV0JzogZGlzdGFuY2VNZXRyaWNzWydCcmF1bi1CbGFucXVldCddLFxuICAgICdDb3NpbmUnOiBkaXN0YW5jZU1ldHJpY3NbJ0Nvc2luZSddLFxuICAgICdLdWxjenluc2tpJzogZGlzdGFuY2VNZXRyaWNzWydLdWxjenluc2tpJ10sXG4gICAgJ01jLUNvbm5hdWdoZXknOiBkaXN0YW5jZU1ldHJpY3NbJ01jLUNvbm5hdWdoZXknXSxcbiAgICAnUm9nb3QtR29sZGJlcmcnOiBkaXN0YW5jZU1ldHJpY3NbJ1JvZ290LUdvbGRiZXJnJ10sXG4gICAgJ1J1c3NlbCc6IGRpc3RhbmNlTWV0cmljc1snUnVzc2VsJ10sXG4gICAgJ1Nva2FsJzogZGlzdGFuY2VNZXRyaWNzWydTb2thbCddLFxuICB9LFxufTtcblxuZXhwb3J0IGNvbnN0IE1ldHJpY1RvRGF0YVR5cGU6IFN0cmluZ0RpY3Rpb25hcnkgPSBPYmplY3Qua2V5cyhBdmFpbGFibGVNZXRyaWNzKVxuICAucmVkdWNlKChyZXQ6IFN0cmluZ0RpY3Rpb25hcnksIGtleSkgPT4ge1xuICAgIGZvciAoY29uc3QgdmFsIG9mIE9iamVjdC5rZXlzKEF2YWlsYWJsZU1ldHJpY3Nba2V5IGFzIEF2YWlsYWJsZURhdGFUeXBlc10pKSB7XG4gICAgICByZXRbdmFsIGFzIEF2YWlsYWJsZURhdGFUeXBlc10gPSBrZXk7XG4gICAgfVxuICAgIHJldHVybiByZXQ7XG4gIH0sIHt9KTtcblxuZXhwb3J0IHR5cGUgQXZhaWxhYmxlRGF0YVR5cGVzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3M7XG5leHBvcnQgdHlwZSBTdHJpbmdNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ1N0cmluZyddO1xuZXhwb3J0IHR5cGUgQml0QXJyYXlNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ0JpdEFycmF5J107XG5leHBvcnQgdHlwZSBWZWN0b3JNZXRyaWNzID0ga2V5b2YgdHlwZW9mIEF2YWlsYWJsZU1ldHJpY3NbJ1ZlY3RvciddO1xuZXhwb3J0IHR5cGUgS25vd25NZXRyaWNzID0gU3RyaW5nTWV0cmljcyB8IEJpdEFycmF5TWV0cmljcyB8IFZlY3Rvck1ldHJpY3M7XG5cbmV4cG9ydCB0eXBlIFZhbGlkVHlwZXMgPSB7ZGF0YTogc3RyaW5nW10sIG1ldHJpYzogU3RyaW5nTWV0cmljc30gfCB7ZGF0YTogVmVjdG9yW10sIG1ldHJpYzogVmVjdG9yTWV0cmljc30gfFxuICAgICAgICAgICAgICAgICAgICAgICAgIHtkYXRhOiBCaXRBcnJheVtdLCBtZXRyaWM6IEJpdEFycmF5TWV0cmljc307XG5cbmV4cG9ydCBmdW5jdGlvbiBpc1N0cmluZ01ldHJpYyhuYW1lOiBLbm93bk1ldHJpY3MpIHtcbiAgcmV0dXJuIE1ldHJpY1RvRGF0YVR5cGVbbmFtZV0gPT0gJ1N0cmluZyc7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBpc0JpdEFycmF5TWV0cmljKG5hbWU6IEtub3duTWV0cmljcykge1xuICByZXR1cm4gTWV0cmljVG9EYXRhVHlwZVtuYW1lXSA9PSAnQml0QXJyYXknO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNWZWN0b3JNZXRyaWMobmFtZTogS25vd25NZXRyaWNzKSB7XG4gIHJldHVybiBNZXRyaWNUb0RhdGFUeXBlW25hbWVdID09ICdWZWN0b3InO1xufVxuXG4vKiogVW5pZmllZCBjbGFzcyBpbXBsZW1lbnRpbmcgZGlmZmVyZW50IHN0cmluZyBtZWFzdXJlcy4gKi9cbmV4cG9ydCBjbGFzcyBNZWFzdXJlIHtcbiAgcHJvdGVjdGVkIG1ldGhvZDogS25vd25NZXRyaWNzO1xuICBwcm90ZWN0ZWQgZGF0YVR5cGU6IEF2YWlsYWJsZURhdGFUeXBlcztcblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiBpbnN0YW5jZSBvZiBNZWFzdXJlIHdpdGggLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gbWV0aG9kIE1ldGhvZCB0byBjYWxjdWxhdGUgZGlzdGFuY2UgYmV0d2VlbiBzdHJpbmdzLlxuICAgKiBAbWVtYmVyb2YgTWVhc3VyZXJcbiAgICovXG4gIGNvbnN0cnVjdG9yKG1ldGhvZDogS25vd25NZXRyaWNzKSB7XG4gICAgdGhpcy5tZXRob2QgPSBtZXRob2Q7XG4gICAgdGhpcy5kYXRhVHlwZSA9IE1ldHJpY1RvRGF0YVR5cGVbbWV0aG9kXSBhcyBBdmFpbGFibGVEYXRhVHlwZXM7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBjdXN0b20gc3RyaW5nIGRpc3RhbmNlIGZ1bmN0aW9uIHNwZWNpZmllZC5cbiAgICogQHJldHVybiB7RGlzdGFuY2VNZXRyaWN9IENhbGxiYWNrIG9mIHRoZSBtZWFzdXJlIGNob3Nlbi5cbiAgICogQG1lbWJlcm9mIE1lYXN1cmVyXG4gICAqL1xuICBwdWJsaWMgZ2V0TWVhc3VyZSgpOiBEaXN0YW5jZU1ldHJpYyB7XG4gICAgY29uc3QgZGljdDoge1trZXk6IHN0cmluZ106IHtba2V5Mjogc3RyaW5nXTogRGlzdGFuY2VNZXRyaWN9fSA9IEF2YWlsYWJsZU1ldHJpY3M7XG4gICAgcmV0dXJuIGRpY3RbdGhpcy5kYXRhVHlwZV1bdGhpcy5tZXRob2RdO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybnMgY3VzdG9tIHN0cmluZyBkaXN0YW5jZSBieSB0aGUgZ2l2ZW4gZGF0YSB0eXBlLlxuICAgKiBAcGFyYW0ge0F2YWlsYWJsZURhdGFUeXBlc30gZGF0YVR5cGUgTWV0cmljJ3MgZGF0YSB0eXBlXG4gICAqIEByZXR1cm4ge3N0cmluZ1tdfSBNZXRyaWMgbmFtZXMgd2hpY2ggZXhwZWN0cyB0aGUgZ2l2ZW4gZGF0YSB0eXBlXG4gICAqIEBtZW1iZXJvZiBNZWFzdXJlclxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBnZXRNZXRyaWNCeURhdGFUeXBlKGRhdGFUeXBlOiBBdmFpbGFibGVEYXRhVHlwZXMpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKEF2YWlsYWJsZU1ldHJpY3NbZGF0YVR5cGVdKTtcbiAgfVxuXG4gIC8qKiBSZXR1cm5zIG1ldHJpYyBuYW1lcyBhdmFpbGFibGUuXG4gICAqIEBtZW1iZXJvZiBNZWFzdXJlclxuICAqL1xuICBzdGF0aWMgZ2V0IGF2YWlsYWJsZU1lYXN1cmVzKCk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gT2JqZWN0LmtleXMoQXZhaWxhYmxlTWV0cmljcyk7XG4gIH1cbn1cbiJdfQ==","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { removeEmptyStringRows } from '@datagrok-libraries/utils/src/dataframe-utils';\nlet zoom = false;\nconst molColumnNames = ['1_seq', '2_seq'];\nconst nonNormalizedDistances = ['Levenshtein'];\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport function getActivityCliffs(df, seqCol, encodedCol, axesNames, scatterTitle, activities, similarity, similarityMetric, methodName, semType, tags, seqSpaceFunc, simMatrixFunc, tooltipFunc, propertyPanelFunc, seqSpaceOptions) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const automaticSimilarityLimit = false;\n        const MIN_SIMILARITY = 80;\n        const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n        const dimensionalityReduceCol = encodedCol !== null && encodedCol !== void 0 ? encodedCol : seqCol;\n        const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n        //@ts-ignore\n        const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n        const { distance, coordinates } = yield seqSpaceFunc({\n            seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name),\n            methodName: methodName,\n            similarityMetric: similarityMetric,\n            embedAxesNames: axesNames,\n            options: seqSpaceOptions\n        });\n        for (const col of coordinates) {\n            const listValues = col.toList();\n            emptyValsIdxs.forEach((ind) => listValues.splice(ind, 0, null));\n            df.columns.add(DG.Column.fromList('double', col.name, listValues));\n        }\n        const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n        const dim = dimensionalityReduceCol.length;\n        const simArr = Array(dim - 1);\n        if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n            yield simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n        else\n            getSimilaritiesFromDistances(dim, distance, simArr);\n        const optSimilarityLimit = initialSimilarityLimit;\n        const simVals = [];\n        const saliVals = [];\n        const n1 = [];\n        const n2 = [];\n        const cliffsMolIds = new Set();\n        for (let i = 0; i != dim - 1; ++i) {\n            for (let j = 0; j != dim - 1 - i; ++j) {\n                const sim = simArr[i] ? simArr[i].get(j) : 0;\n                if (sim >= optSimilarityLimit) {\n                    n1.push(i);\n                    n2.push(i + j + 1);\n                    cliffsMolIds.add(i);\n                    cliffsMolIds.add(i + j + 1);\n                    simVals.push(sim);\n                    const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n                    if (sim != 1)\n                        saliVals.push(diff / (1 - sim));\n                    else\n                        saliVals.push(Infinity);\n                }\n            }\n        }\n        const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n        const saliMin = Math.min(...saliValsWithoutInfinity);\n        const saliMax = Math.max(...saliValsWithoutInfinity);\n        const saliOpacityCoef = 0.8 / (saliMax - saliMin);\n        const neighboursCount = new Array(dim).fill(0);\n        const similarityCount = new Array(dim).fill(0);\n        const saliCount = new Array(dim).fill(0);\n        for (let i = 0; i != n1.length; ++i) {\n            neighboursCount[n1[i]] += 1;\n            neighboursCount[n2[i]] += 1;\n            similarityCount[n1[i]] += simVals[i];\n            similarityCount[n2[i]] += simVals[i];\n            if (saliVals[i] != Infinity) {\n                if (activities.get(n1[i]) > activities.get(n2[i]))\n                    saliCount[n1[i]] += saliVals[i];\n                else\n                    saliCount[n2[i]] += saliVals[i];\n            }\n        }\n        const sali = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n        df.columns.add(sali);\n        const cliffsColName = addCliffsBooleanCol(df, cliffsMolIds);\n        const view = grok.shell.getTableView(df.name);\n        view.grid.columns.byName(cliffsColName).visible = false;\n        const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n            xColumnName: axesNames[0],\n            yColumnName: axesNames[1],\n            size: sali.name,\n            color: activities.name,\n            showXSelector: false,\n            showYSelector: false,\n            showSizeSelector: false,\n            showColorSelector: false,\n            markerMinSize: 5,\n            markerMaxSize: 25,\n            title: scatterTitle\n        }));\n        const canvas = sp.getInfo()['canvas'];\n        const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n        const cashedLinesData = {};\n        let acc;\n        linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n            const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n            const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n            sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n            sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n            sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n            linesDfGrid.invalidate();\n            if (line)\n                setTimeout(() => { updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc); }, 1000);\n        });\n        linesRes.linesDf.onSelectionChanged.subscribe((_) => {\n            if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n                const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n                line.selected = !line.selected;\n                if (!line.selected)\n                    df.selection.setAll(false);\n            }\n            linesRes.lines.forEach((l) => {\n                if (l.selected)\n                    l.mols.forEach((m) => df.selection.set(m, true));\n            });\n            linesDfGrid.invalidate();\n        });\n        const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n        linesDfGrid.onCellClick.subscribe(() => {\n            zoom = true;\n        });\n        const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n            grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n        });\n        editSpLinkStyle(listCliffsLink, sp, 10);\n        const filterCliffsButton = ui.switchInput(`Show only cliffs`, false);\n        filterCliffsButton.onChanged(() => {\n            filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n        });\n        editSpLinkStyle(filterCliffsButton.root, sp, 30);\n        let timer;\n        canvas.addEventListener('mousemove', function (event) {\n            clearTimeout(timer);\n            timer = global.setTimeout(function () {\n                const line = checkCursorOnLine(event, canvas, linesRes.lines);\n                if (line && df.mouseOverRowIdx === -1) {\n                    ui.tooltip.show(tooltipFunc({ cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities }), event.clientX, event.clientY);\n                }\n            }, 500);\n        });\n        canvas.addEventListener('mousedown', function (event) {\n            const line = checkCursorOnLine(event, canvas, linesRes.lines);\n            if (line && df.mouseOverRowIdx === -1) {\n                if (event.ctrlKey) {\n                    line.selected = !line.selected;\n                    linesRes.linesDf.selection.set(line.id, line.selected);\n                    if (!line.selected)\n                        df.selection.setAll(false);\n                    linesRes.lines.forEach((l) => {\n                        if (l.selected)\n                            l.mols.forEach((m) => df.selection.set(m, true));\n                    });\n                }\n                else {\n                    if (linesRes.linesDf.currentRowIdx !== line.id) {\n                        linesRes.linesDf.currentRowIdx = line.id;\n                        df.currentRowIdx = line.mols[0];\n                        df.selection.set(0, !linesRes.lines[0].selected);\n                        df.selection.set(0, linesRes.lines[0].selected);\n                    }\n                }\n                const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n                linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n            }\n        });\n        sp.onEvent('d4-before-draw-scene')\n            .subscribe((_) => {\n            const lines = renderLines(sp, axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n            if (zoom) {\n                const currentLine = lines[linesRes.linesDf.currentRowIdx];\n                setTimeout(() => {\n                    const { zoomLeft, zoomRight, zoomTop, zoomBottom } = getZoomCoordinates(sp.viewport.width, sp.viewport.height, sp.dataFrame.get(axesNames[0], currentLine.mols[0]), sp.dataFrame.get(axesNames[1], currentLine.mols[0]), sp.dataFrame.get(axesNames[0], currentLine.mols[1]), sp.dataFrame.get(axesNames[1], currentLine.mols[1]));\n                    sp.zoom(zoomLeft, zoomTop, zoomRight, zoomBottom);\n                }, 300);\n                zoom = false;\n            }\n            filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n        });\n        sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n        acc = createPopertyPanel();\n        return sp;\n    });\n}\nfunction editSpLinkStyle(el, sp, topMargin) {\n    el.style.position = 'absolute';\n    el.style.top = `${topMargin}px`;\n    el.style.right = '10px';\n    sp.root.append(el);\n}\nfunction createPopertyPanel() {\n    const acc = ui.accordion();\n    const accIcon = ui.element('i');\n    accIcon.className = 'grok-icon svg-icon svg-view-layout';\n    acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n    acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n    grok.shell.o = acc.root;\n    return acc;\n}\nfunction updatePropertyPanel(df, acc, cashedData, line, seqCol, activities, sali, propPanelFunc) {\n    const panel = acc.getPane('Cliff Details');\n    ui.empty(panel.root);\n    const panelElement = propPanelFunc({ cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali });\n    panel.root.append(panelElement);\n    setTimeout(() => {\n        grok.shell.o = acc.root;\n    }, 500);\n}\nfunction getZoomCoordinates(W0, H0, x1, y1, x2, y2) {\n    const W1 = Math.abs(x1 - x2);\n    const H1 = Math.abs(y1 - y2);\n    const scaleW = W0 / W1;\n    const scaleH = H0 / H1;\n    const scale = Math.min(scaleW, scaleH);\n    const W2 = (W0 / scale) * 5;\n    const H2 = (H0 / scale) * 5;\n    const left = x1 < x2 ? x1 : x2;\n    const top = y1 > y2 ? y1 : y2;\n    const zoomLeft = (left + W1 / 2) - W2 / 2;\n    const zoomRight = zoomLeft + W2;\n    const zoomTop = (top - H1 / 2) + H2 / 2;\n    const zoomBottom = zoomTop - H2;\n    return { zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom };\n}\nfunction checkCursorOnLine(event, canvas, lines) {\n    const rect = canvas.getBoundingClientRect();\n    const x = event.clientX - rect.left;\n    const y = event.clientY - rect.top;\n    let closestLine = null;\n    let minDist = 0;\n    for (const line of lines) {\n        const dist = Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n            Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n        if ((!minDist && dist < 2) || dist < minDist) {\n            minDist = dist;\n            closestLine = line;\n        }\n    }\n    return closestLine;\n}\nfunction renderLines(sp, xAxis, yAxis, linesRes, saliVals, saliOpacityCoef, saliMin) {\n    const lines = linesRes.lines;\n    const canvas = sp.getInfo()['canvas'];\n    const ctx = canvas.getContext('2d');\n    const x = sp.dataFrame.columns.byName(xAxis);\n    const y = sp.dataFrame.columns.byName(yAxis);\n    for (let i = 0; i < lines.length; i++) {\n        const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n        const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n        lines[i].a = [pointFrom.x, pointFrom.y];\n        lines[i].b = [pointTo.x, pointTo.y];\n        const line = new Path2D();\n        line.moveTo(lines[i].a[0], lines[i].a[1]);\n        const color = lines[i].selected ? '255,255,0' : '0,128,0';\n        const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin) * saliOpacityCoef;\n        ctx.strokeStyle = `rgba(${color},${opacity})`;\n        ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n        line.lineTo(lines[i].b[0], lines[i].b[1]);\n        ctx.stroke(line);\n    }\n    return lines;\n}\nfunction createLines(n1, n2, seq, activities, saliVals, simVals, semType, tags) {\n    const lines = new Array(n1.length).fill(null);\n    for (let i = 0; i < n1.length; i++) {\n        const num1 = n1[i];\n        const num2 = n2[i];\n        lines[i] = ({ id: i, mols: [num1, num2], selected: false, a: [], b: [] });\n    }\n    const linesDf = DG.DataFrame.create(lines.length);\n    molColumnNames.forEach((it, idx) => {\n        linesDf.columns.addNewString(it).init((i) => seq.get(lines[i].mols[idx]));\n        setTags(linesDf.col(it), tags);\n        linesDf.col(it).semType = semType;\n    });\n    linesDf.columns.addNewFloat('act_diff')\n        .init((i) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n    linesDf.columns.addNewInt('line_index').init((i) => i);\n    linesDf.columns.addNewFloat('sali').init((i) => saliVals[i]);\n    linesDf.columns.addNewFloat('sim').init((i) => simVals[i]);\n    return { lines, linesDf };\n}\nfunction setTags(col, tags) {\n    Object.keys(tags).forEach(tag => {\n        col.tags[tag] = tags[tag];\n    });\n}\nexport function getSimilaritiesMarix(dim, seqCol, dfSeq, simArr, simFunc) {\n    return __awaiter(this, void 0, void 0, function* () {\n        for (let i = 0; i != dim - 1; ++i) {\n            const mol = seqCol.get(i);\n            dfSeq.rows.removeAt(0, 1, false);\n            simArr[i] = (yield simFunc(dfSeq.col('seq'), mol));\n        }\n        return simArr;\n    });\n}\nexport function getSimilaritiesFromDistances(dim, distances, simArr) {\n    for (let i = 0; i < dim - 1; ++i) {\n        const similarityArr = new Array(dim - i - 1).fill(0);\n        for (let j = i + 1; j < dim; ++j) {\n            similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n        }\n        simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n    }\n    return simArr;\n}\nexport function addCliffsBooleanCol(df, ids) {\n    const colname = 'containsCliff';\n    const colNameInd = df.columns.names().filter((it) => it.includes(colname)).length + 1;\n    const newColName = `${colname}_${colNameInd}`;\n    df.columns.addNewBool(newColName).init((i) => ids.has(i));\n    return newColName;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"activity-cliffs.js","sourceRoot":"","sources":["activity-cliffs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as grok from 'datagrok-api/grok';\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { Matrix } from '@datagrok-libraries/utils/src/type-declarations';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport {removeEmptyStringRows} from '@datagrok-libraries/utils/src/dataframe-utils'\n\nexport interface ILine {\n  id: number;\n  mols: number[];\n  selected: boolean;\n  a: number[]; // [x, y]\n  b: number[]; // [x, y]\n}\n\ninterface IRenderedLines {\n  lines: ILine[];\n  linesDf: DG.DataFrame;\n}\n\nexport interface ISequenceSpaceParams {\n  seqCol: DG.Column,\n  methodName: string,\n  similarityMetric: string,\n  embedAxesNames: string[],\n  options?: any\n}\n\nexport interface ISequenceSpaceResult {\n  distance: Matrix;\n  coordinates: DG.ColumnList;\n}\n\nexport interface ITooltipAndPanelParams {\n  cashedData: any,\n  line: ILine,\n  df: DG.DataFrame,\n  seqCol: DG.Column,\n  activityCol: DG.Column,\n  sali?: number\n}\n\nlet zoom = false;\n\nconst molColumnNames = ['1_seq', '2_seq'];\n\nconst nonNormalizedDistances = ['Levenshtein'];\n\n// Searches for activity cliffs in a chemical dataset by selected cutoff\nexport async function getActivityCliffs(\n    df: DG.DataFrame, \n    seqCol: DG.Column,\n    encodedCol: DG.Column | null,\n    axesNames: string[],\n    scatterTitle: string,\n    activities: DG.Column, \n    similarity: number,\n    similarityMetric: string, \n    methodName: string, \n    semType: string,\n    tags: {[index: string]: string},\n    seqSpaceFunc: (params: ISequenceSpaceParams) => Promise<ISequenceSpaceResult>,\n    simMatrixFunc: (dim: number, seqCol: DG.Column, df: DG.DataFrame, colName: string, simArr: DG.Column[]) => Promise<DG.Column[]>,\n    tooltipFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    propertyPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement,\n    seqSpaceOptions?: any) : Promise<DG.Viewer> {\n\n  const automaticSimilarityLimit = false;\n  const MIN_SIMILARITY = 80;\n\n  const initialSimilarityLimit = automaticSimilarityLimit ? MIN_SIMILARITY : similarity / 100;\n\n  const dimensionalityReduceCol = encodedCol ?? seqCol;\n  const withoutEmptyValues = DG.DataFrame.fromColumns([dimensionalityReduceCol]).clone();\n    //@ts-ignore\n  const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, dimensionalityReduceCol);\n\n  const {distance, coordinates} = await seqSpaceFunc({\n    seqCol: withoutEmptyValues.col(dimensionalityReduceCol.name)!,\n    methodName: methodName,\n    similarityMetric: similarityMetric,\n    embedAxesNames: axesNames,\n    options: seqSpaceOptions\n  });\n\n  for (const col of coordinates) {\n      const listValues = col.toList();\n      emptyValsIdxs.forEach((ind: number) => listValues.splice(ind, 0, null));\n      df.columns.add(DG.Column.fromList('double', col.name, listValues));\n  }\n\n  const dfSeq = DG.DataFrame.fromColumns([DG.Column.fromList('string', 'seq', dimensionalityReduceCol.toList())]);\n  const dim = dimensionalityReduceCol.length;\n  const simArr: DG.Column[] = Array(dim - 1);\n\n  if (!distance || emptyValsIdxs.length !== 0 || nonNormalizedDistances.includes(similarityMetric))\n    await simMatrixFunc(dim, dimensionalityReduceCol, dfSeq, 'seq', simArr);\n  else\n    getSimilaritiesFromDistances(dim, distance, simArr);\n\n  const optSimilarityLimit = initialSimilarityLimit;\n\n  const simVals: number[] = [];\n  const saliVals: number[] = [];\n  const n1: number[] = [];\n  const n2: number[] = [];\n  const cliffsMolIds = new Set<number>();\n\n  for (let i = 0; i != dim - 1; ++i) {\n    for (let j = 0; j != dim - 1 - i; ++j) {\n      const sim: number = simArr[i] ? simArr[i].get(j) : 0;\n\n      if (sim >= optSimilarityLimit) {\n        n1.push(i);\n        n2.push(i + j + 1);\n        cliffsMolIds.add(i);\n        cliffsMolIds.add(i + j + 1);\n        simVals.push(sim);\n        const diff = Math.abs(activities.get(i) - activities.get(i + j + 1));\n        if (sim != 1)\n          saliVals.push(diff / (1 - sim));\n        else\n          saliVals.push(Infinity);\n      }\n    }\n  }\n\n  const saliValsWithoutInfinity = saliVals.filter(it => it !== Infinity);\n  const saliMin = Math.min(...saliValsWithoutInfinity);\n  const saliMax = Math.max(...saliValsWithoutInfinity);\n  const saliOpacityCoef = 0.8/(saliMax - saliMin);\n\n\n  const neighboursCount = new Array(dim).fill(0);\n  const similarityCount = new Array(dim).fill(0);\n  const saliCount = new Array(dim).fill(0);\n\n  for (let i = 0; i != n1.length; ++i) {\n    neighboursCount[n1[i]] += 1;\n    neighboursCount[n2[i]] += 1;\n    similarityCount[n1[i]] += simVals[i];\n    similarityCount[n2[i]] += simVals[i];\n    if (saliVals[i] != Infinity) {\n      if (activities.get(n1[i]) > activities.get(n2[i]))\n        saliCount[n1[i]] += saliVals[i];\n      else\n        saliCount[n2[i]] += saliVals[i];\n    }\n  }\n\n  const sali: DG.Column = DG.Column.fromList('double', `sali_${axesNames[0].substring(axesNames[0].lastIndexOf('_'))}`, saliCount);\n\n  df.columns.add(sali);\n  const cliffsColName = addCliffsBooleanCol(df, cliffsMolIds);\n\n  const view = grok.shell.getTableView(df.name);\n  view.grid.columns.byName(cliffsColName)!.visible = false;\n  const sp = view.addViewer(DG.Viewer.scatterPlot(df, {\n    xColumnName: axesNames[0],\n    yColumnName: axesNames[1],\n    size: sali.name,\n    color: activities.name,\n    showXSelector: false,\n    showYSelector: false,\n    showSizeSelector: false,\n    showColorSelector: false,\n    markerMinSize: 5,\n    markerMaxSize: 25,\n    title: scatterTitle\n  })) as DG.ScatterPlotViewer;\n\n  const canvas = (sp.getInfo() as any)['canvas'];\n  const linesRes = createLines(n1, n2, seqCol, activities, saliVals, simVals, semType, tags);\n\n  const cashedLinesData: any = {};\n  let acc: DG.Accordion;\n\n  linesRes.linesDf.onCurrentCellChanged.subscribe(() => {\n    const currentMolIdx = linesRes.linesDf.currentCol && linesRes.linesDf.currentCol.name === '2_seq' ? 1 : 0;\n    const line = linesRes.linesDf.currentRowIdx !== -1 ? linesRes.lines[linesRes.linesDf.currentRowIdx] : null;\n    sp.dataFrame.currentRowIdx = line ? line.mols[currentMolIdx] : -1;\n    sp.dataFrame.selection.set(0, !linesRes.lines[0].selected);\n    sp.dataFrame.selection.set(0, linesRes.lines[0].selected);\n    linesDfGrid.invalidate();\n    if (line)\n      setTimeout(() => {updatePropertyPanel(df, acc, cashedLinesData, line, seqCol, activities, linesRes.linesDf.get('sali', line.id), propertyPanelFunc)}, 1000);      \n  });\n\n  linesRes.linesDf.onSelectionChanged.subscribe((_: any) => {\n    if (linesRes.linesDf.mouseOverRowIdx !== -1) {\n      const line = linesRes.lines[linesRes.linesDf.mouseOverRowIdx];\n      line.selected = !line.selected;\n      if (!line.selected)\n        df.selection.setAll(false);\n    }\n    linesRes.lines.forEach((l) => {\n      if (l.selected)\n        l.mols.forEach((m) => df.selection.set(m, true));\n    });\n    linesDfGrid.invalidate();\n  });\n\n  const linesDfGrid = linesRes.linesDf.plot.grid().sort(['sali'], [false]);\n\n  linesDfGrid.onCellClick.subscribe(() => {\n    zoom = true;\n  });\n\n  const listCliffsLink = ui.button(`${linesRes.linesDf.rowCount} cliffs`, () => {\n    grok.shell.dockManager.dock(linesDfGrid.root, 'down', null, 'Activity cliffs', 0.2);\n  });\n  editSpLinkStyle(listCliffsLink, sp, 10);\n\n  const filterCliffsButton = ui.switchInput(`Show only cliffs`, false);\n  filterCliffsButton.onChanged(()=> {\n    filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n  });\n  editSpLinkStyle(filterCliffsButton.root, sp, 30);\n\n  let timer: NodeJS.Timeout;\n  canvas.addEventListener('mousemove', function (event: MouseEvent) {\n    clearTimeout(timer);\n    timer = global.setTimeout(function () {\n      const line = checkCursorOnLine(event, canvas, linesRes.lines);\n      if (line && df.mouseOverRowIdx === -1) {\n          ui.tooltip.show(tooltipFunc({cashedData: cashedLinesData, line: line, df: df, seqCol: seqCol, activityCol: activities}), event.clientX, event.clientY);\n      }\n    }, 500);\n  });\n\n  canvas.addEventListener('mousedown', function(event: MouseEvent) {\n    const line = checkCursorOnLine(event, canvas, linesRes.lines);\n    if (line && df.mouseOverRowIdx === -1) {\n      if (event.ctrlKey) {\n        line.selected = !line.selected;\n        linesRes.linesDf.selection.set(line.id, line.selected);\n\n        if (!line.selected)\n          df.selection.setAll(false);\n\n        linesRes.lines.forEach((l) => {\n          if (l.selected)\n            l.mols.forEach((m) => df.selection.set(m, true));\n        });\n      } else {\n        if (linesRes.linesDf.currentRowIdx !== line.id) {\n          linesRes.linesDf.currentRowIdx = line.id;\n          df.currentRowIdx = line.mols[0];\n          df.selection.set(0, !linesRes.lines[0].selected);\n          df.selection.set(0, linesRes.lines[0].selected);\n        }\n      }\n      const order = linesRes.linesDf.getSortedOrder(linesDfGrid.sortByColumns, linesDfGrid.sortTypes);\n      linesDfGrid.scrollToCell('seq_1', order.indexOf(line.id));\n    }\n  });\n\n  sp.onEvent('d4-before-draw-scene')\n    .subscribe((_: any) => {\n      const lines = renderLines(sp,\n        axesNames[0], axesNames[1], linesRes, saliVals, saliOpacityCoef, saliMin);\n      if (zoom) {\n        const currentLine = lines[linesRes.linesDf.currentRowIdx];\n        setTimeout(()=> {\n          const {zoomLeft, zoomRight, zoomTop, zoomBottom} = getZoomCoordinates(\n            sp.viewport.width,\n            sp.viewport.height,\n            sp.dataFrame.get(axesNames[0], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[0]),\n            sp.dataFrame.get(axesNames[0], currentLine.mols[1]),\n            sp.dataFrame.get(axesNames[1], currentLine.mols[1])\n          )        \n          sp.zoom(zoomLeft,\n            zoomTop,\n            zoomRight,\n            zoomBottom);\n        }, 300);\n        zoom = false;\n      }\n      filterCliffsButton.value ? df.rows.filter((row) => row[cliffsColName] === true) : df.filter.setAll(true);\n    });\n\n  sp.addProperty('similarityLimit', 'double', optSimilarityLimit);\n  acc = createPopertyPanel();\n  return sp;\n}\n\nfunction editSpLinkStyle(el: HTMLElement, sp: DG.ScatterPlotViewer, topMargin: number){\n  el.style.position = 'absolute';\n  el.style.top = `${topMargin}px`;\n  el.style.right = '10px';\n  sp.root.append(el);\n}\n\nfunction createPopertyPanel(): DG.Accordion {\n  const acc = ui.accordion();\n  const accIcon = ui.element('i');\n  accIcon.className = 'grok-icon svg-icon svg-view-layout';\n  acc.addTitle(ui.span([accIcon, ui.label(`Activity cliffs`)]));\n  acc.addPane('Cliff Details', () => ui.divText('Cliff has not been selected'), true);\n  grok.shell.o = acc.root;\n  return acc;\n}\n\nfunction updatePropertyPanel(\n  df: DG.DataFrame,\n  acc: DG.Accordion,\n  cashedData: any,\n  line: ILine, \n  seqCol: DG.Column, \n  activities: DG.Column,\n  sali: number,\n  propPanelFunc: (params: ITooltipAndPanelParams) => HTMLElement){\n  const panel = acc.getPane('Cliff Details');\n  ui.empty(panel.root);\n  const panelElement = propPanelFunc({cashedData: cashedData, line: line, df: df, seqCol: seqCol, activityCol: activities, sali: sali});\n  panel.root.append(panelElement);\n  setTimeout(() => {\n    grok.shell.o = acc.root;\n  }, 500);\n}\n\nfunction getZoomCoordinates(W0: number, H0: number, x1: number, y1: number, x2: number, y2: number) {\n  const W1 = Math.abs(x1 - x2);\n  const H1 = Math.abs(y1 - y2);\n  const scaleW = W0 / W1;\n  const scaleH = H0 / H1;\n  const scale = Math.min(scaleW, scaleH);\n  const W2 = (W0 / scale) * 5;\n  const H2 = (H0 / scale) * 5;\n  const left = x1 < x2 ? x1 : x2;\n  const top = y1 > y2 ? y1 : y2;\n  const zoomLeft = (left + W1 / 2) - W2 / 2;\n  const zoomRight = zoomLeft + W2;\n  const zoomTop = (top - H1 / 2) + H2 / 2;\n  const zoomBottom = zoomTop - H2;\n  return {zoomLeft: zoomLeft, zoomRight: zoomRight, zoomTop: zoomTop, zoomBottom: zoomBottom};\n}\n\nfunction checkCursorOnLine(event: any, canvas: any, lines: ILine[]): ILine | null {\n  const rect = canvas.getBoundingClientRect();\n  const x = event.clientX - rect.left;\n  const y = event.clientY - rect.top;\n  let closestLine = null;\n  let minDist = 0;\n  for (const line of lines) {\n    const dist =\n      Math.abs(Math.hypot(line.a[0] - x, line.a[1] - y) +\n      Math.hypot(line.b[0] - x, line.b[1] - y) - Math.hypot(line.a[0] - line.b[0], line.a[1] - line.b[1]));\n    if ((!minDist && dist < 2) || dist < minDist) {\n      minDist = dist;\n      closestLine = line;\n    }\n  }\n  return closestLine;\n}\n\nfunction renderLines(sp: DG.ScatterPlotViewer,\n  xAxis: string, yAxis: string, linesRes: IRenderedLines, saliVals: number[], saliOpacityCoef: number, saliMin: number): ILine [] {\n  const lines = linesRes.lines;\n  const canvas = (sp.getInfo() as {[index: string] : any})['canvas'];\n  const ctx = canvas.getContext('2d') as CanvasRenderingContext2D;\n  const x = sp.dataFrame!.columns.byName(xAxis);\n  const y = sp.dataFrame!.columns.byName(yAxis);\n  for (let i = 0; i < lines.length; i++) {\n    const pointFrom = sp.worldToScreen(x.get(lines[i].mols[0]), y.get(lines[i].mols[0]));\n    const pointTo = sp.worldToScreen(x.get(lines[i].mols[1]), y.get(lines[i].mols[1]));\n    lines[i].a = [pointFrom.x, pointFrom.y];\n    lines[i].b = [pointTo.x, pointTo.y];\n    const line = new Path2D();\n    line.moveTo(lines[i].a[0], lines[i].a[1]);\n    const color = lines[i].selected ? '255,255,0' : '0,128,0';\n    const opacity = saliVals[i] === Infinity ? 1 : 0.2 + (saliVals[i] - saliMin)*saliOpacityCoef;\n    ctx.strokeStyle = `rgba(${color},${opacity})`;\n    ctx.lineWidth = lines[i].id === linesRes.linesDf.currentRowIdx ? 3 : 1;\n    line.lineTo(lines[i].b[0], lines[i].b[1]);\n    ctx.stroke(line);\n  }\n  return lines;\n}\n\nfunction createLines(\n  n1: number[], \n  n2: number[], \n  seq: DG.Column, \n  activities: DG.Column, \n  saliVals: number[],\n  simVals: number[],\n  semType: string,\n  tags: {[index: string]: string}) : IRenderedLines {\n  const lines: ILine[] = new Array(n1.length).fill(null);\n  for (let i = 0; i < n1.length; i++) {\n    const num1 = n1[i];\n    const num2 = n2[i];\n    lines[i] = ({id: i, mols: [num1, num2], selected: false, a: [], b: []});\n  }\n  const linesDf = DG.DataFrame.create(lines.length);\n  molColumnNames.forEach((it, idx) => {\n    linesDf.columns.addNewString(it).init((i: number) => seq.get(lines[i].mols[idx]));\n    setTags(linesDf.col(it)!, tags);\n    linesDf.col(it)!.semType = semType;\n  })\n  linesDf.columns.addNewFloat('act_diff')\n    .init((i: number) => Math.abs(activities.get(lines[i].mols[0]) - activities.get(lines[i].mols[1])));\n  linesDf.columns.addNewInt('line_index').init((i: number) => i);\n  linesDf.columns.addNewFloat('sali').init((i: number) => saliVals[i]);\n  linesDf.columns.addNewFloat('sim').init((i: number) => simVals[i]);\n  return {lines, linesDf};\n}\n\nfunction setTags(col: DG.Column, tags: {[index: string]: string}) {\n  Object.keys(tags).forEach(tag => {\n    col.tags[tag] = tags[tag];\n  })\n}\n\nexport async function getSimilaritiesMarix(\n    dim: number, \n    seqCol: DG.Column, \n    dfSeq: DG.DataFrame, \n    simArr: DG.Column[],\n    simFunc: (col: DG.Column, mol: string) => Promise<DG.Column | null>\n    )\n  : Promise<DG.Column[]> {\n  for (let i = 0; i != dim - 1; ++i) {\n    const mol = seqCol.get(i);\n    dfSeq.rows.removeAt(0, 1, false);\n    simArr[i] = (await simFunc(dfSeq.col('seq')!, mol))!;\n  }\n  return simArr;\n}\n\n\nexport function getSimilaritiesFromDistances(dim: number, distances: Matrix, simArr: DG.Column[])\n  : DG.Column[] {\n  for (let i = 0; i < dim - 1; ++i) {\n    const similarityArr = new Array(dim - i - 1).fill(0);\n    for (let j = i + 1; j < dim; ++j) {\n      similarityArr[j - i - 1] = getSimilarityFromDistance(distances[i][j]);\n    }\n    simArr[i] = DG.Column.fromFloat32Array('similarity', Float32Array.from(similarityArr));\n  }\n  return simArr;\n}\n\nexport function addCliffsBooleanCol(df: DG.DataFrame, ids: Set<number>): string {\n  const colname = 'containsCliff';\n  const colNameInd = df.columns.names().filter((it: string) => it.includes(colname)).length + 1;\n  const newColName = `${colname}_${colNameInd}`;\n  df.columns.addNewBool(newColName).init((i) => ids.has(i));\n  return newColName;\n}\n\n"]}","export function createDimensinalityReducingWorker(dataMetric, method, options) {\n    return new Promise(function (resolve) {\n        const worker = new Worker(new URL('./dimensionality-reducer', import.meta.url));\n        worker.postMessage({\n            columnData: dataMetric.data,\n            method: method,\n            measure: dataMetric.metric,\n            options: options,\n        });\n        worker.onmessage = ({ data: { distance, embedding } }) => {\n            resolve({ distance: distance, embedding: embedding });\n        };\n    });\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGltZW5zaW9uYWxpdHktcmVkdWNpbmctd29ya2VyLWNyZWF0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJkaW1lbnNpb25hbGl0eS1yZWR1Y2luZy13b3JrZXItY3JlYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpQkEsTUFBTSxVQUFVLGlDQUFpQyxDQUFDLFVBQXNCLEVBQUUsTUFBYyxFQUNsRixPQUFhO0lBRWpCLE9BQU8sSUFBSSxPQUFPLENBQUMsVUFBUyxPQUFPO1FBQ2pDLE1BQU0sTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksR0FBRyxDQUFDLDBCQUEwQixFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUNoRixNQUFNLENBQUMsV0FBVyxDQUFDO1lBQ2pCLFVBQVUsRUFBRSxVQUFVLENBQUMsSUFBSTtZQUMzQixNQUFNLEVBQUUsTUFBTTtZQUNkLE9BQU8sRUFBRSxVQUFVLENBQUMsTUFBTTtZQUMxQixPQUFPLEVBQUUsT0FBTztTQUNqQixDQUFDLENBQUM7UUFDSCxNQUFNLENBQUMsU0FBUyxHQUFHLENBQUMsRUFBQyxJQUFJLEVBQUUsRUFBQyxRQUFRLEVBQUUsU0FBUyxFQUFDLEVBQUMsRUFBRSxFQUFFO1lBQ3BELE9BQU8sQ0FBQyxFQUFDLFFBQVEsRUFBRSxRQUFRLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBQyxDQUFDLENBQUM7UUFDckQsQ0FBQyxDQUFDO0lBQ0osQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWF0cml4IH0gZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy91dGlscy9zcmMvdHlwZS1kZWNsYXJhdGlvbnMnO1xuaW1wb3J0IHtWYWxpZFR5cGVzfSBmcm9tICcuLi90eXBlZC1tZXRyaWNzJztcblxuLyoqXG4gKiBBIHdvcmtlciB0byBwZXJmb3JtIGRpbWVuc2lvbmFsaXR5IHJlZHVjdGlvbi5cbiAqXG4gKiBAcGFyYW0ge1ZhbGlkVHlwZXN9IGRhdGFNZXRyaWMgVGhlIGRhdGEgdG8gcHJvY2Vzcy5cbiAqIEBwYXJhbSB7c3RyaW5nfSBtZXRob2QgQSBtZXRob2Qgb2YgZGltZW5zaW9uYWxpdHkgcmVkdWN0aW9uLlxuICogQHBhcmFtIG9wdGlvbnMgLSBrZXktdmFsdWUgcGFpcnNcbiAqIEBwYXJhbSByZXR1cm5EaXN0YW5jZU1hdHJpeFxuICogQHJldHVybiB7UHJvbWlzZTxJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQ+fSBSZXN1bHRpbmcgZW1iZWRkaW5nIGFuZCBkaXN0YW5jZSBtYXRyaXguXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVJlZHVjZURpbWVuc2lvbmFsaXR5UmVzdWx0IHtcbiAgZGlzdGFuY2U6IE1hdHJpeDtcbiAgZW1iZWRkaW5nOiBNYXRyaXg7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVEaW1lbnNpbmFsaXR5UmVkdWNpbmdXb3JrZXIoZGF0YU1ldHJpYzogVmFsaWRUeXBlcywgbWV0aG9kOiBzdHJpbmcsXG4gICAgICBvcHRpb25zPzogYW55KTogUHJvbWlzZTxJUmVkdWNlRGltZW5zaW9uYWxpdHlSZXN1bHQ+IHtcblxuICByZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24ocmVzb2x2ZSkge1xuICAgIGNvbnN0IHdvcmtlciA9IG5ldyBXb3JrZXIobmV3IFVSTCgnLi9kaW1lbnNpb25hbGl0eS1yZWR1Y2VyJywgaW1wb3J0Lm1ldGEudXJsKSk7XG4gICAgd29ya2VyLnBvc3RNZXNzYWdlKHtcbiAgICAgIGNvbHVtbkRhdGE6IGRhdGFNZXRyaWMuZGF0YSxcbiAgICAgIG1ldGhvZDogbWV0aG9kLFxuICAgICAgbWVhc3VyZTogZGF0YU1ldHJpYy5tZXRyaWMsXG4gICAgICBvcHRpb25zOiBvcHRpb25zLFxuICAgIH0pO1xuICAgIHdvcmtlci5vbm1lc3NhZ2UgPSAoe2RhdGE6IHtkaXN0YW5jZSwgZW1iZWRkaW5nfX0pID0+IHtcbiAgICAgcmVzb2x2ZSh7ZGlzdGFuY2U6IGRpc3RhbmNlLCBlbWJlZGRpbmc6IGVtYmVkZGluZ30pO1xuICAgIH07XG4gIH0pO1xufVxuIl19","import * as DG from 'datagrok-api/dg';\nimport * as sha256 from 'fast-sha256';\n/**\n * For columns of string type. Checks whether column contains empty values and removes corresponding rows in case user selects to remove.\n *\n */\nexport function removeEmptyStringRows(table, col) {\n    const cats = col.categories;\n    const emptyRawInd = cats.map((val, ind) => !val ? ind : null).filter((it) => it !== null);\n    const rawData = [...col.getRawData()];\n    const emptyRawsIndexes = [];\n    let removedRowsCounter = 0;\n    for (let i = 0; i < table.rowCount; i++) {\n        if (emptyRawInd.includes(rawData[i])) {\n            table.rows.removeAt(i - removedRowsCounter);\n            emptyRawsIndexes.push(i);\n            removedRowsCounter += 1;\n        }\n    }\n    return emptyRawsIndexes;\n}\nexport function hashDataFrame(table, names) {\n    names !== null && names !== void 0 ? names : (names = table.columns.names());\n    const hasher = new sha256.Hash();\n    const order = table.getSortedOrder(names);\n    const encoder = new TextEncoder();\n    for (const name of names) {\n        const column = table.columns.byName(name);\n        const dataArray = column.getRawData();\n        const isString = column.type == DG.TYPE.STRING;\n        const cats = column.categories;\n        for (let i = 0; i < dataArray.length; i++) {\n            if (isString) {\n                const data = cats[dataArray[order[i]]];\n                hasher.update(encoder.encode(data));\n            }\n            else {\n                const data = dataArray[order[i]];\n                hasher.update(Uint8Array.from([data]));\n            }\n        }\n    }\n    return hasher.digest();\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWZyYW1lLXV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZGF0YWZyYW1lLXV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDdEMsT0FBTyxLQUFLLE1BQU0sTUFBTSxhQUFhLENBQUM7QUFDdEM7OztHQUdHO0FBQ0gsTUFBTSxVQUFVLHFCQUFxQixDQUFDLEtBQW1CLEVBQUUsR0FBYztJQUN2RSxNQUFNLElBQUksR0FBRyxHQUFHLENBQUMsVUFBVSxDQUFDO0lBQzVCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRSxDQUFDLEVBQUUsS0FBSyxJQUFJLENBQUMsQ0FBQztJQUMxRixNQUFNLE9BQU8sR0FBRyxDQUFDLEdBQUcsR0FBRyxDQUFDLFVBQVUsRUFBRSxDQUFDLENBQUM7SUFDdEMsTUFBTSxnQkFBZ0IsR0FBRyxFQUFFLENBQUM7SUFDNUIsSUFBSSxrQkFBa0IsR0FBRyxDQUFDLENBQUM7SUFDM0IsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEtBQUssQ0FBQyxRQUFRLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDdkMsSUFBSSxXQUFXLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFO1lBQ3BDLEtBQUssQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDO1lBQzVDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN6QixrQkFBa0IsSUFBSSxDQUFDLENBQUM7U0FDekI7S0FDRjtJQUNELE9BQU8sZ0JBQWdCLENBQUM7QUFDMUIsQ0FBQztBQUVELE1BQU0sVUFBVSxhQUFhLENBQUMsS0FBbUIsRUFBRSxLQUFnQjtJQUNqRSxLQUFLLGFBQUwsS0FBSyxjQUFMLEtBQUssSUFBTCxLQUFLLEdBQUssS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsRUFBQztJQUNoQyxNQUFNLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNqQyxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsY0FBYyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzFDLE1BQU0sT0FBTyxHQUFHLElBQUksV0FBVyxFQUFFLENBQUM7SUFDbEMsS0FBSyxNQUFNLElBQUksSUFBSSxLQUFLLEVBQUU7UUFDeEIsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDMUMsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ3RDLE1BQU0sUUFBUSxHQUFHLE1BQU0sQ0FBQyxJQUFJLElBQUksRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUM7UUFDL0MsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQztRQUMvQixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsU0FBUyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUN6QyxJQUFJLFFBQVEsRUFBRTtnQkFDWixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBQ3ZDLE1BQU0sQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO2FBQ3JDO2lCQUFNO2dCQUNMLE1BQU0sSUFBSSxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztnQkFDakMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO2FBQ3hDO1NBQ0Y7S0FDRjtJQUNELE9BQU8sTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO0FBQ3pCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuaW1wb3J0ICogYXMgc2hhMjU2IGZyb20gJ2Zhc3Qtc2hhMjU2Jztcbi8qKlxuICogRm9yIGNvbHVtbnMgb2Ygc3RyaW5nIHR5cGUuIENoZWNrcyB3aGV0aGVyIGNvbHVtbiBjb250YWlucyBlbXB0eSB2YWx1ZXMgYW5kIHJlbW92ZXMgY29ycmVzcG9uZGluZyByb3dzIGluIGNhc2UgdXNlciBzZWxlY3RzIHRvIHJlbW92ZS5cbiAqXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiByZW1vdmVFbXB0eVN0cmluZ1Jvd3ModGFibGU6IERHLkRhdGFGcmFtZSwgY29sOiBERy5Db2x1bW4pOiBudW1iZXJbXSB7XG4gIGNvbnN0IGNhdHMgPSBjb2wuY2F0ZWdvcmllcztcbiAgY29uc3QgZW1wdHlSYXdJbmQgPSBjYXRzLm1hcCgodmFsLCBpbmQpID0+ICF2YWwgPyBpbmQgOiBudWxsKS5maWx0ZXIoKGl0KSA9PiBpdCAhPT0gbnVsbCk7XG4gIGNvbnN0IHJhd0RhdGEgPSBbLi4uY29sLmdldFJhd0RhdGEoKV07XG4gIGNvbnN0IGVtcHR5UmF3c0luZGV4ZXMgPSBbXTtcbiAgbGV0IHJlbW92ZWRSb3dzQ291bnRlciA9IDA7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgdGFibGUucm93Q291bnQ7IGkrKykge1xuICAgIGlmIChlbXB0eVJhd0luZC5pbmNsdWRlcyhyYXdEYXRhW2ldKSkge1xuICAgICAgdGFibGUucm93cy5yZW1vdmVBdChpIC0gcmVtb3ZlZFJvd3NDb3VudGVyKTtcbiAgICAgIGVtcHR5UmF3c0luZGV4ZXMucHVzaChpKTtcbiAgICAgIHJlbW92ZWRSb3dzQ291bnRlciArPSAxO1xuICAgIH1cbiAgfVxuICByZXR1cm4gZW1wdHlSYXdzSW5kZXhlcztcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGhhc2hEYXRhRnJhbWUodGFibGU6IERHLkRhdGFGcmFtZSwgbmFtZXM/OiBzdHJpbmdbXSk6IFVpbnQ4QXJyYXkge1xuICBuYW1lcyA/Pz0gdGFibGUuY29sdW1ucy5uYW1lcygpO1xuICBjb25zdCBoYXNoZXIgPSBuZXcgc2hhMjU2Lkhhc2goKTtcbiAgY29uc3Qgb3JkZXIgPSB0YWJsZS5nZXRTb3J0ZWRPcmRlcihuYW1lcyk7XG4gIGNvbnN0IGVuY29kZXIgPSBuZXcgVGV4dEVuY29kZXIoKTtcbiAgZm9yIChjb25zdCBuYW1lIG9mIG5hbWVzKSB7XG4gICAgY29uc3QgY29sdW1uID0gdGFibGUuY29sdW1ucy5ieU5hbWUobmFtZSk7XG4gICAgY29uc3QgZGF0YUFycmF5ID0gY29sdW1uLmdldFJhd0RhdGEoKTtcbiAgICBjb25zdCBpc1N0cmluZyA9IGNvbHVtbi50eXBlID09IERHLlRZUEUuU1RSSU5HO1xuICAgIGNvbnN0IGNhdHMgPSBjb2x1bW4uY2F0ZWdvcmllcztcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGRhdGFBcnJheS5sZW5ndGg7IGkrKykge1xuICAgICAgaWYgKGlzU3RyaW5nKSB7XG4gICAgICAgIGNvbnN0IGRhdGEgPSBjYXRzW2RhdGFBcnJheVtvcmRlcltpXV1dO1xuICAgICAgICBoYXNoZXIudXBkYXRlKGVuY29kZXIuZW5jb2RlKGRhdGEpKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IGRhdGEgPSBkYXRhQXJyYXlbb3JkZXJbaV1dO1xuICAgICAgICBoYXNoZXIudXBkYXRlKFVpbnQ4QXJyYXkuZnJvbShbZGF0YV0pKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcmV0dXJuIGhhc2hlci5kaWdlc3QoKTtcbn1cbiJdfQ==","/**\n * Generates single random float from 0 to range.\n *\n * @export\n * @param {number} range Max generating value.\n * @return {number} A random float generated.\n */\nexport function randomFloat(range) {\n    return Math.random() * range;\n}\n/**\n * Generates single random integer from 0 to range.\n *\n * @export\n * @param {number} range Max generating value.\n * @return {number} A random integer generated.\n */\nexport function randomInt(range) {\n    return Math.floor(randomFloat(range));\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFuZG9tLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicmFuZG9tLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxXQUFXLENBQUMsS0FBYTtJQUN2QyxPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxLQUFLLENBQUM7QUFDL0IsQ0FBQztBQUVEOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxTQUFTLENBQUMsS0FBYTtJQUNyQyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFDeEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVzIHNpbmdsZSByYW5kb20gZmxvYXQgZnJvbSAwIHRvIHJhbmdlLlxuICpcbiAqIEBleHBvcnRcbiAqIEBwYXJhbSB7bnVtYmVyfSByYW5nZSBNYXggZ2VuZXJhdGluZyB2YWx1ZS5cbiAqIEByZXR1cm4ge251bWJlcn0gQSByYW5kb20gZmxvYXQgZ2VuZXJhdGVkLlxuICovXG5leHBvcnQgZnVuY3Rpb24gcmFuZG9tRmxvYXQocmFuZ2U6IG51bWJlcik6IG51bWJlciB7XG4gIHJldHVybiBNYXRoLnJhbmRvbSgpICogcmFuZ2U7XG59XG5cbi8qKlxuICogR2VuZXJhdGVzIHNpbmdsZSByYW5kb20gaW50ZWdlciBmcm9tIDAgdG8gcmFuZ2UuXG4gKlxuICogQGV4cG9ydFxuICogQHBhcmFtIHtudW1iZXJ9IHJhbmdlIE1heCBnZW5lcmF0aW5nIHZhbHVlLlxuICogQHJldHVybiB7bnVtYmVyfSBBIHJhbmRvbSBpbnRlZ2VyIGdlbmVyYXRlZC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJhbmRvbUludChyYW5nZTogbnVtYmVyKTogbnVtYmVyIHtcbiAgcmV0dXJuIE1hdGguZmxvb3IocmFuZG9tRmxvYXQocmFuZ2UpKTtcbn1cbiJdfQ==","import { randomInt } from './random';\nexport const similarityMetric = {\n    'Tanimoto': tanimotoSimilarity,\n    'Dice': diceSimilarity,\n    'Asymmetric': asymmetricSimilarity,\n    'Braun-Blanquet': braunBlanquetSimilarity,\n    'Cosine': cosineSimilarity,\n    'Kulczynski': kulczynskiSimilarity,\n    'Mc-Connaughey': mcConnaugheySimilarity,\n    'Rogot-Goldberg': rogotGoldbergSimilarity,\n    'Russel': russelSimilarity,\n    'Sokal': sokalSimilarity,\n    'Hamming': hammingSimilarity,\n    'Euclidean': euclideanSimilarity,\n};\nexport const distanceMetrics = {\n    'Tanimoto': tanimotoDistance,\n    'Dice': diceDistance,\n    'Asymmetric': asymmetricDistance,\n    'Braun-Blanquet': braunBlanquetDistance,\n    'Cosine': cosineDistance,\n    'Kulczynski': kulczynskiDistance,\n    'Mc-Connaughey': mcConnaugheyDistance,\n    'Rogot-Goldberg': rogotGoldbergDistance,\n    'Russel': russelDistance,\n    'Sokal': sokalDistance,\n    'Hamming': hammingDistance,\n    'Euclidean': euclideanDistance,\n};\nexport const CHEM_SIMILARITY_METRICS = ['Tanimoto', 'Dice', 'Cosine', 'Hamming', 'Euclidean'];\nexport function tanimotoSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    if (total == 0)\n        return 1.0;\n    const common = x.andWithCountBits(y, true);\n    return common / (total - common);\n}\nexport function tanimotoDistance(x, y) {\n    return getDistanceFromSimilarity(tanimotoSimilarity(x, y));\n}\nexport function diceSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    if (total == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return 2 * common / total;\n}\nexport function diceDistance(x, y) {\n    return getDistanceFromSimilarity(diceSimilarity(x, y));\n}\nexport function cosineSimilarity(x, y) {\n    const total = x.trueCount() * y.trueCount();\n    if (total == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / Math.sqrt(total);\n}\nexport function cosineDistance(x, y) {\n    return getDistanceFromSimilarity(cosineSimilarity(x, y));\n}\nexport function euclideanSimilarity(x, y) {\n    return getSimilarityFromDistance(euclideanDistance(x, y));\n}\nexport function euclideanDistance(x, y) {\n    return Math.sqrt(x.trueCount() + y.trueCount() - 2 * x.andWithCountBits(y, true));\n}\nexport function hammingSimilarity(x, y) {\n    return getSimilarityFromDistance(hammingDistance(x, y));\n}\nexport function hammingDistance(x, y) {\n    return x.trueCount() + y.trueCount() - 2 * x.andWithCountBits(y, true);\n}\nexport function sokalSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const common = x.andWithCountBits(y, true);\n    return common / (2 * total - 3 * common);\n}\nexport function sokalDistance(x, y) {\n    return getDistanceFromSimilarity(sokalSimilarity(x, y));\n}\nexport function kulczynskiSimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const totalProd = x.trueCount() * y.trueCount();\n    if (totalProd == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return (common * total) / (2 * totalProd);\n}\nexport function kulczynskiDistance(x, y) {\n    return getDistanceFromSimilarity(kulczynskiSimilarity(x, y));\n}\nexport function mcConnaugheySimilarity(x, y) {\n    const total = x.trueCount() + y.trueCount();\n    const totalProd = x.trueCount() * y.trueCount();\n    if (totalProd == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return (common * total - totalProd) / totalProd;\n}\nexport function mcConnaugheyDistance(x, y) {\n    return getDistanceFromSimilarity(mcConnaugheySimilarity(x, y));\n}\nexport function asymmetricSimilarity(x, y) {\n    const min = Math.min(x.trueCount(), y.trueCount());\n    if (min == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / min;\n}\nexport function asymmetricDistance(x, y) {\n    return getDistanceFromSimilarity(asymmetricSimilarity(x, y));\n}\nexport function braunBlanquetSimilarity(x, y) {\n    const max = Math.max(x.trueCount(), y.trueCount());\n    if (max == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / max;\n}\nexport function braunBlanquetDistance(x, y) {\n    return getDistanceFromSimilarity(braunBlanquetSimilarity(x, y));\n}\nexport function russelSimilarity(x, y) {\n    if (x.length == 0)\n        return 0.0;\n    const common = x.andWithCountBits(y, true);\n    return common / x.length;\n}\nexport function russelDistance(x, y) {\n    return getDistanceFromSimilarity(russelSimilarity(x, y));\n}\nexport function rogotGoldbergSimilarity(x, y) {\n    const common = x.andWithCountBits(y, true);\n    const total = x.countBits(true) + y.countBits(true);\n    const len = x.length;\n    const diff = len - total + common;\n    if ((common == len) || (diff == len))\n        return 1.0;\n    else\n        return common / total + diff / (2 * len - total);\n}\nexport function rogotGoldbergDistance(x, y) {\n    return getDistanceFromSimilarity(rogotGoldbergSimilarity(x, y));\n}\nexport function getSimilarityFromDistance(distance) {\n    return 1 / (1 + distance);\n}\nexport function getDistanceFromSimilarity(similarity) {\n    return 1 / similarity - 1;\n}\nexport function getDiverseSubset(length, n, dist) {\n    function maxBy(values, orderBy) {\n        let maxValue = null;\n        let maxOrderBy = null;\n        for (const element of values) {\n            const elementOrderBy = orderBy(element);\n            if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n                maxValue = element;\n                maxOrderBy = elementOrderBy;\n            }\n        }\n        return maxValue;\n    }\n    const subset = [randomInt(length - 1)];\n    const complement = new Set();\n    for (let i = 0; i < length; ++i) {\n        if (!subset.includes(i))\n            complement.add(i);\n    }\n    while (subset.length < n) {\n        const idx = maxBy(complement.values(), (i) => Math.min.apply(Math, subset.map(function (val, index) {\n            return dist(i, val);\n        })));\n        if (idx != null) {\n            subset.push(idx);\n            complement.delete(idx);\n        }\n    }\n    return subset;\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"similarity-metrics.js","sourceRoot":"","sources":["similarity-metrics.ts"],"names":[],"mappings":"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","sourcesContent":["import BitArray from './bit-array';\nimport {randomInt} from './random';\n\nexport const similarityMetric: {[name: string]: (x: BitArray, y: BitArray) => number} = {\n  'Tanimoto': tanimotoSimilarity,\n  'Dice': diceSimilarity,\n  'Asymmetric': asymmetricSimilarity,\n  'Braun-Blanquet': braunBlanquetSimilarity,\n  'Cosine': cosineSimilarity,\n  'Kulczynski': kulczynskiSimilarity,\n  'Mc-Connaughey': mcConnaugheySimilarity,\n  'Rogot-Goldberg': rogotGoldbergSimilarity,\n  'Russel': russelSimilarity,\n  'Sokal': sokalSimilarity,\n  'Hamming': hammingSimilarity,\n  'Euclidean': euclideanSimilarity,\n};\n\nexport const distanceMetrics: {[name: string]: (x: BitArray, y: BitArray) => number} = {\n  'Tanimoto': tanimotoDistance,\n  'Dice': diceDistance,\n  'Asymmetric': asymmetricDistance,\n  'Braun-Blanquet': braunBlanquetDistance,\n  'Cosine': cosineDistance,\n  'Kulczynski': kulczynskiDistance,\n  'Mc-Connaughey': mcConnaugheyDistance,\n  'Rogot-Goldberg': rogotGoldbergDistance,\n  'Russel': russelDistance,\n  'Sokal': sokalDistance,\n  'Hamming': hammingDistance,\n  'Euclidean': euclideanDistance,\n};\n\nexport const CHEM_SIMILARITY_METRICS = ['Tanimoto', 'Dice', 'Cosine', 'Hamming', 'Euclidean'];\n\nexport function tanimotoSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  if (total == 0)\n    return 1.0;\n  const common = x.andWithCountBits(y, true);\n  return common / (total - common);\n}\n\nexport function tanimotoDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(tanimotoSimilarity(x, y));\n}\n\nexport function diceSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  if (total == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return 2 * common / total;\n}\n\nexport function diceDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(diceSimilarity(x, y));\n}\n\nexport function cosineSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() * y.trueCount();\n  if (total == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / Math.sqrt(total);\n}\n\nexport function cosineDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(cosineSimilarity(x, y));\n}\n\nexport function euclideanSimilarity(x: BitArray, y: BitArray): number {\n  return getSimilarityFromDistance(euclideanDistance(x, y));\n}\n\nexport function euclideanDistance(x: BitArray, y: BitArray): number {\n  return Math.sqrt(x.trueCount() + y.trueCount() - 2*x.andWithCountBits(y, true));\n}\n\nexport function hammingSimilarity(x: BitArray, y: BitArray): number {\n  return getSimilarityFromDistance(hammingDistance(x, y));\n}\n\nexport function hammingDistance(x: BitArray, y: BitArray): number {\n  return x.trueCount() + y.trueCount() - 2*x.andWithCountBits(y, true);\n}\n\nexport function sokalSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const common = x.andWithCountBits(y, true);\n  return common / (2 * total - 3 * common);\n}\n\nexport function sokalDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(sokalSimilarity(x, y));\n}\n\nexport function kulczynskiSimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const totalProd = x.trueCount() * y.trueCount();\n  if (totalProd == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return (common * total) / (2 * totalProd);\n}\n\nexport function kulczynskiDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(kulczynskiSimilarity(x, y));\n}\n\nexport function mcConnaugheySimilarity(x: BitArray, y: BitArray): number {\n  const total = x.trueCount() + y.trueCount();\n  const totalProd = x.trueCount() * y.trueCount();\n  if (totalProd == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return (common * total - totalProd) / totalProd;\n}\n\nexport function mcConnaugheyDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(mcConnaugheySimilarity(x, y));\n}\n\nexport function asymmetricSimilarity(x: BitArray, y: BitArray): number {\n  const min = Math.min(x.trueCount(), y.trueCount());\n  if (min == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / min;\n}\n\nexport function asymmetricDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(asymmetricSimilarity(x, y));\n}\n\nexport function braunBlanquetSimilarity(x: BitArray, y: BitArray): number {\n  const max = Math.max(x.trueCount(), y.trueCount());\n  if (max == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / max;\n}\n\nexport function braunBlanquetDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(braunBlanquetSimilarity(x, y));\n}\n\nexport function russelSimilarity(x: BitArray, y: BitArray): number {\n  if (x.length == 0)\n    return 0.0;\n  const common = x.andWithCountBits(y, true);\n  return common / x.length;\n}\n\nexport function russelDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(russelSimilarity(x, y));\n}\n\nexport function rogotGoldbergSimilarity(x: BitArray, y: BitArray): number {\n  const common = x.andWithCountBits(y, true);\n  const total = x.countBits(true) + y.countBits(true);\n  const len = x.length;\n  const diff = len - total + common;\n  if ((common == len) || (diff == len)) return 1.0;\n  else return common / total + diff / (2 * len - total);\n}\n\nexport function rogotGoldbergDistance(x: BitArray, y: BitArray): number {\n  return getDistanceFromSimilarity(rogotGoldbergSimilarity(x, y));\n}\n\nexport function getSimilarityFromDistance(distance: number) {\n  return 1 / ( 1 + distance);\n}\n\nexport function getDistanceFromSimilarity(similarity: number) {\n  return 1 / similarity - 1;\n}\n\nexport function getDiverseSubset(length: number, n: number, dist: (i1: number, i2: number) => number) {\n  function maxBy(values: IterableIterator<number>, orderBy: (i: number) => number) {\n    let maxValue = null;\n    let maxOrderBy = null;\n\n    for (const element of values) {\n      const elementOrderBy = orderBy(element);\n      if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n        maxValue = element;\n        maxOrderBy = elementOrderBy;\n      }\n    }\n    return maxValue;\n  }\n\n  const subset = [randomInt(length - 1)];\n  const complement = new Set();\n\n  for (let i = 0; i < length; ++i) {\n    if (!subset.includes(i))\n      complement.add(i);\n  }\n\n  while (subset.length < n) {\n    const idx = maxBy(\n      complement.values() as IterableIterator<number>,\n      (i) => Math.min.apply(Math, subset.map(function(val, index) {\n        return dist(i, val);\n      })));\n    if (idx != null) {\n      subset.push(idx);\n      complement.delete(idx);\n    }\n  }\n  return subset;\n}\n"]}","/**\n * Denotes a vector of floating poit values.\n *\n * @export\n * @class Vector\n * @extends {Float32Array}\n */\nexport class Vector extends Float32Array {\n}\n/**\n * Denotes a two-dimensional matrix.\n *\n * @export\n * @class Matrix\n * @extends {Array<Vector>}\n */\nexport class Matrix extends Array {\n}\n/**\n * Denotes cartesian coordinates.\n *\n * @export\n * @class Coordinates\n * @extends {Matrix}\n */\nexport class Coordinates extends Matrix {\n}\n/**\n * Denotes an array of arbitrary-typed vectors.\n *\n * @export\n * @class Vectors\n * @extends {Array<any>}\n */\nexport class Vectors extends Array {\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZS1kZWNsYXJhdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ0eXBlLWRlY2xhcmF0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sTUFBTyxTQUFRLFlBQVk7Q0FBRztBQUUzQzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sTUFBTyxTQUFRLEtBQWE7Q0FBRztBQUU1Qzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sV0FBWSxTQUFRLE1BQU07Q0FBRztBQUUxQzs7Ozs7O0dBTUc7QUFDSCxNQUFNLE9BQU8sT0FBUSxTQUFRLEtBQVU7Q0FBRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogRGVub3RlcyBhIHZlY3RvciBvZiBmbG9hdGluZyBwb2l0IHZhbHVlcy5cbiAqXG4gKiBAZXhwb3J0XG4gKiBAY2xhc3MgVmVjdG9yXG4gKiBAZXh0ZW5kcyB7RmxvYXQzMkFycmF5fVxuICovXG5leHBvcnQgY2xhc3MgVmVjdG9yIGV4dGVuZHMgRmxvYXQzMkFycmF5IHt9XG5cbi8qKlxuICogRGVub3RlcyBhIHR3by1kaW1lbnNpb25hbCBtYXRyaXguXG4gKlxuICogQGV4cG9ydFxuICogQGNsYXNzIE1hdHJpeFxuICogQGV4dGVuZHMge0FycmF5PFZlY3Rvcj59XG4gKi9cbmV4cG9ydCBjbGFzcyBNYXRyaXggZXh0ZW5kcyBBcnJheTxWZWN0b3I+IHt9XG5cbi8qKlxuICogRGVub3RlcyBjYXJ0ZXNpYW4gY29vcmRpbmF0ZXMuXG4gKlxuICogQGV4cG9ydFxuICogQGNsYXNzIENvb3JkaW5hdGVzXG4gKiBAZXh0ZW5kcyB7TWF0cml4fVxuICovXG5leHBvcnQgY2xhc3MgQ29vcmRpbmF0ZXMgZXh0ZW5kcyBNYXRyaXgge31cblxuLyoqXG4gKiBEZW5vdGVzIGFuIGFycmF5IG9mIGFyYml0cmFyeS10eXBlZCB2ZWN0b3JzLlxuICpcbiAqIEBleHBvcnRcbiAqIEBjbGFzcyBWZWN0b3JzXG4gKiBAZXh0ZW5kcyB7QXJyYXk8YW55Pn1cbiAqL1xuZXhwb3J0IGNsYXNzIFZlY3RvcnMgZXh0ZW5kcyBBcnJheTxhbnk+IHt9XG5cbi8qKlxuICogRGVub3RlcyBhIGRpY3Rpb25hcnkgY29udGFpbmluZyBmdW5jdGlvbiBvcHRpb25zLlxuICpcbiAqIEBleHBvcnRcbiAqIEB0eXBlIE9wdGlvbnNcbiAqL1xuZXhwb3J0IHR5cGUgT3B0aW9ucyA9IHtbbmFtZTogc3RyaW5nXTogYW55fTtcblxuLyoqXG4gKiBEZW5vdGVzIGN1c3RvbSBkaXN0YW5jZSBtZXRyaWMgYmV0d2VlbiB0aGUgdHdvIGdpdmVuIHZlY3RvcnMuXG4gKlxuICogQGV4cG9ydFxuICogQHR5cGUgRGlzdGFuY2VNZXRyaWNcbiAqIEBwYXJhbSB7YW55fSB2MSBUaGUgZmlyc3QgdmVjdG9yLlxuICogQHBhcmFtIHthbnl9IHYyIFRoZSBzZWNvbmQgdmVjdG9yLlxuICogQHJldHVybiB7bnVtYmVyfSBEaXN0YW5jZSBiZXR3ZWVuIHRoZXNlIHR3byB2ZWN0b3JzLlxuICovXG5leHBvcnQgdHlwZSBEaXN0YW5jZU1ldHJpYyA9ICh2MTogYW55LCB2MjogYW55KSA9PiAobnVtYmVyKTtcblxuLyoqXG4gKiBEZW5vdGVzIGEgc2ltcGxlIHN0cmluZyB0byBzdHJpbmcgZGljdGlvbmFyeS5cbiAqXG4gKiBAZXhwb3J0XG4gKiBAdHlwZSBTdHJpbmdEaWN0aW9uYXJ5XG4gKi9cbmV4cG9ydCB0eXBlIFN0cmluZ0RpY3Rpb25hcnkgPSB7W2tleTogc3RyaW5nXTogc3RyaW5nfTtcbiJdfQ==","import { Vector } from './type-declarations';\nimport { randomFloat, randomInt } from './random';\n/**\n * Asserts a condition by throwing an Error.\n *\n * @export\n * @param {boolean} [condition=false] Condition to assert.\n * @param {string} [message='Assertion error.'] Message to output.\n * @throws {Error}\n */\nexport function assert(condition = false, message = 'Assertion error.') {\n    if (!condition)\n        throw new Error(message);\n}\n/**\n * Creates new two-dimensional array and fills it with the value given.\n *\n * @param {number} dimension1 The first dimension of the coordinates (number of rows).\n * @param {number} dimension2 The second dimension of the coordinates (number of columns).\n * @param {number} [fill=0] A value to fill the coordinates with.\n * @return {Coordinates} A two-dimensional filled with the value given.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nfunction initCoordinates(dimension1, dimension2, fill = 0) {\n    return new Array(dimension1).fill(fill).map(() => (new Vector(dimension2).fill(fill)));\n}\n/**\n * Transpose matrix.\n *\n * @export\n * @param {Matrix} matrix The matrix to be transposed.\n * @return {Matrix} Transposed matrix.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nexport function transposeMatrix(matrix) {\n    return new Array(matrix[0].length).fill(0)\n        .map((_, i) => (new Vector(matrix.length).fill(0).map((_, j) => (matrix[j][i]))));\n}\n/**\n * Adds two vectors with the second one to be multiplied by the given ratio.\n *\n * @export\n * @param {Vector} p The first vector to add.\n * @param {Vector} q The second vector to add.\n * @param {number} [multiplier=1] A multiplier to be used before the second vector is added.\n * @return {Vector} New vector contained the result of operation p+multiplier*q.\n */\nexport function vectorAdd(p, q, multiplier = 1) {\n    const nItems = p.length;\n    assert(nItems == q.length, 'Vector lengths do not match.');\n    const total = new Vector(nItems);\n    for (let i = 0; i < p.length; ++i)\n        total[i] = p[i] + multiplier * q[i];\n    return total;\n}\n/**\n * Sums the vector's items.\n *\n * @param {Vector} v The vector to be summed.\n * @return {number} The vector's items sum.\n */\nfunction itemsSum(v) {\n    let total = 0;\n    for (let i = 0; i < v.length; ++i)\n        total += v[i];\n    return total;\n}\n/**\n * Suqares the vector's items.\n *\n * @param {Vector} v The vector to square.\n * @return {Vector} A new vector containing the original's items squared.\n */\nfunction vectorSquare(v) {\n    const nItems = v.length;\n    const total = new Vector(nItems);\n    for (let i = 0; i < v.length; ++i)\n        total[i] = v[i] * v[i];\n    return total;\n}\nexport function vectorLength(v) {\n    let sqrSum = 0;\n    for (let i = 0; i < v.length; i++)\n        sqrSum += v[i] * v[i];\n    return Math.sqrt(sqrSum);\n}\nexport function vectorDotProduct(v1, v2) {\n    if (v1.length != v2.length)\n        throw Error('The dimensionality of the vectors must match');\n    let prod = 0;\n    for (let i = 0; i < v1.length; i++)\n        prod += v1[i] * v2[i];\n    return prod;\n}\n/**\n * Creates a matrix filled with random floating point values.\n *\n * @export\n * @param {number} dimension1 The first dimension of the matrix.\n * @param {number} dimension2 The second dimension of the matrix.\n * @param {number} [scale=1.] Max value given by random generator.\n * @return {Matrix} A new matrix filled with random floating point  values.\n */\nexport function fillRandomMatrix(dimension1, dimension2, scale = 1.) {\n    const matrix = initCoordinates(dimension1, dimension2);\n    for (let i = 0; i < dimension1; ++i) {\n        for (let j = 0; j < dimension2; ++j)\n            matrix[i][j] = randomFloat(scale);\n    }\n    return matrix;\n}\n/**\n * Calculates Euclidean distance between two vectors.\n *\n * @export\n * @param {Vector} p The first vector.\n * @param {Vector} q The second vector.\n * @return {number} Euclidean distance between the given vectors.\n */\nexport function calculateEuclideanDistance(p, q) {\n    const diff = vectorAdd(p, q, -1);\n    const sqdiff = vectorSquare(diff);\n    const sqdiffSumm = itemsSum(sqdiff);\n    return Math.sqrt(sqdiffSumm);\n}\n/**\n * Creates a distance matrix using a custom distance function.\n *\n * @export\n * @param {Vectors} data Input vectors to calculate distances.\n * @param {DistanceMetric} distance Custom distance function.\n * @return {Matrix} Calculated custom distance matrix.\n */\nexport function calcDistanceMatrix(data, distance) {\n    const nItems = data.length;\n    const matrix = initCoordinates(nItems, nItems, 0);\n    for (let i = 0; i < nItems; ++i) {\n        for (let j = i + 1; j < nItems; ++j) {\n            const d = (data[i] == null) || (data[j] == null) ? 0 : distance(data[i], data[j]);\n            matrix[i][j] = matrix[j][i] = d;\n        }\n    }\n    return matrix;\n}\n/** Generates array from a range [begin; end] or [begin; end) if endExclusive. **/\nexport function genRange(begin, end, endExclusive = false) {\n    const nItems = end - begin + (endExclusive ? 0 : 1);\n    const series = new Int32Array(nItems);\n    for (let i = 0; i < nItems; ++i)\n        series[i] = begin + i;\n    return series;\n}\n/**\n * Returns order of values as if they are sorted.\n *\n * @export\n * @param {any[]} values Input array.\n * @param {boolean} [reverse=false] Whether to return reversed order.\n * @return {number[]} The order computed.\n */\nexport function argSort(values, reverse = false) {\n    const sortfn = reverse ? (a, b) => (b[0] - a[0]) : (a, b) => (a[0] - b[0]);\n    const decor = (v, i) => [v, i]; // set index to value\n    const undecor = (a) => a[1]; // leave only index\n    const _argsort = (arr) => arr.map(decor).sort(sortfn).map(undecor);\n    return _argsort(values);\n}\n/**\n * Returns the indexes of the most diverse objects according to the dist function\n * @param {number} length total number of objects\n * @param {number} n number of diverse elements to find\n * @param {(i1: number, i2: number) => number} dist a function which calculates distance between\n *                                                  two objects using their indexes\n * @returns {number[]} The indexes of the most diverse objects\n */\nexport function getDiverseSubset(length, n, dist) {\n    function maxBy(values, orderBy) {\n        let maxValue = null;\n        let maxOrderBy = null;\n        for (const element of values) {\n            const elementOrderBy = orderBy(element);\n            if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n                maxValue = element;\n                maxOrderBy = elementOrderBy;\n            }\n        }\n        return maxValue;\n    }\n    const subset = [randomInt(length - 1)];\n    const complement = new Set();\n    for (let i = 0; i < length; ++i) {\n        if (!subset.includes(i))\n            complement.add(i);\n    }\n    while (subset.length < n) {\n        const idx = maxBy(complement.values(), (i) => Math.min.apply(Math, subset.map(function (val, index) {\n            return dist(i, val);\n        })));\n        if (idx) {\n            subset.push(idx);\n            complement.delete(idx);\n        }\n    }\n    return subset;\n}\n/**\n * Returns normalized vector\n * @param {Vector} data numerical array\n */\nexport function normalize(data) {\n    let mean = 0;\n    let std = 0;\n    for (let i = 0; i < data.length; ++i)\n        mean += data[i];\n    mean /= data.length;\n    for (let i = 0; i < data.length; ++i)\n        std += (data[i] - mean) * (data[i] - mean);\n    std = Math.sqrt(std / data.length);\n    for (let i = 0; i < data.length; ++i)\n        data[i] = (data[i] - mean) / std;\n    return data;\n}\n/**\n * Finds set difference between two lists.\n * @param {any[]} a The first list.\n * @param {any[]} b The second list.\n * @return {any[]}\n */\nexport function setDifference(a, b) {\n    const bSet = new Set(b);\n    return Array.from(new Set(a.filter((x) => !bSet.has(x))).values());\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vector-operations.js","sourceRoot":"","sources":["vector-operations.ts"],"names":[],"mappings":"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","sourcesContent":["import {Matrix, Vector, Coordinates, Vectors, DistanceMetric} from './type-declarations';\nimport {randomFloat, randomInt} from './random';\n\n/**\n * Asserts a condition by throwing an Error.\n *\n * @export\n * @param {boolean} [condition=false] Condition to assert.\n * @param {string} [message='Assertion error.'] Message to output.\n * @throws {Error}\n */\nexport function assert(condition: boolean = false, message: string = 'Assertion error.') {\n  if (!condition)\n    throw new Error(message);\n}\n\n/**\n * Creates new two-dimensional array and fills it with the value given.\n *\n * @param {number} dimension1 The first dimension of the coordinates (number of rows).\n * @param {number} dimension2 The second dimension of the coordinates (number of columns).\n * @param {number} [fill=0] A value to fill the coordinates with.\n * @return {Coordinates} A two-dimensional filled with the value given.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nfunction initCoordinates(dimension1: number, dimension2: number, fill: number = 0): Coordinates {\n  return new Array(dimension1).fill(fill).map(() => (new Vector(dimension2).fill(fill)));\n}\n\n/**\n * Transpose matrix.\n *\n * @export\n * @param {Matrix} matrix The matrix to be transposed.\n * @return {Matrix} Transposed matrix.\n * @todo Might be slow since used Array.map. Probably needs performance revision.\n */\nexport function transposeMatrix(matrix: Matrix): Matrix {\n  return new Array(matrix[0].length).fill(0)\n    .map((_, i) => (new Vector(matrix.length).fill(0).map((_, j) => (matrix[j][i]))));\n}\n\n/**\n * Adds two vectors with the second one to be multiplied by the given ratio.\n *\n * @export\n * @param {Vector} p The first vector to add.\n * @param {Vector} q The second vector to add.\n * @param {number} [multiplier=1] A multiplier to be used before the second vector is added.\n * @return {Vector} New vector contained the result of operation p+multiplier*q.\n */\nexport function vectorAdd(p: Vector, q: Vector, multiplier: number = 1): Vector {\n  const nItems = p.length;\n\n  assert(nItems == q.length, 'Vector lengths do not match.');\n\n  const total = new Vector(nItems);\n\n  for (let i = 0; i < p.length; ++i)\n    total[i] = p[i] + multiplier * q[i];\n\n  return total;\n}\n\n/**\n * Sums the vector's items.\n *\n * @param {Vector} v The vector to be summed.\n * @return {number} The vector's items sum.\n */\nfunction itemsSum(v: Vector): number {\n  let total = 0;\n\n  for (let i = 0; i < v.length; ++i)\n    total += v[i];\n\n  return total;\n}\n\n/**\n * Suqares the vector's items.\n *\n * @param {Vector} v The vector to square.\n * @return {Vector} A new vector containing the original's items squared.\n */\nfunction vectorSquare(v: Vector): Vector {\n  const nItems = v.length;\n  const total = new Vector(nItems);\n\n  for (let i = 0; i < v.length; ++i)\n    total[i] = v[i] * v[i];\n\n  return total;\n}\n\nexport function vectorLength(v: Vector): number {\n  let sqrSum: number = 0;\n  for (let i: number = 0; i < v.length; i++)\n    sqrSum += v[i] * v[i];\n  return Math.sqrt(sqrSum);\n}\n\nexport function vectorDotProduct(v1: Vector, v2: Vector): number {\n  if (v1.length != v2.length)\n    throw Error('The dimensionality of the vectors must match');\n  let prod: number = 0;\n  for (let i: number = 0; i < v1.length; i++)\n    prod += v1[i] * v2[i];\n  return prod;\n}\n\n/**\n * Creates a matrix filled with random floating point values.\n *\n * @export\n * @param {number} dimension1 The first dimension of the matrix.\n * @param {number} dimension2 The second dimension of the matrix.\n * @param {number} [scale=1.] Max value given by random generator.\n * @return {Matrix} A new matrix filled with random floating point  values.\n */\nexport function fillRandomMatrix(dimension1: number, dimension2: number, scale: number = 1.): Matrix {\n  const matrix = initCoordinates(dimension1, dimension2);\n\n  for (let i = 0; i < dimension1; ++i) {\n    for (let j = 0; j < dimension2; ++j)\n      matrix[i][j] = randomFloat(scale);\n  }\n  return matrix;\n}\n\n/**\n * Calculates Euclidean distance between two vectors.\n *\n * @export\n * @param {Vector} p The first vector.\n * @param {Vector} q The second vector.\n * @return {number} Euclidean distance between the given vectors.\n */\nexport function calculateEuclideanDistance(p: Vector, q: Vector): number {\n  const diff = vectorAdd(p, q, -1);\n  const sqdiff = vectorSquare(diff);\n  const sqdiffSumm = itemsSum(sqdiff);\n  return Math.sqrt(sqdiffSumm);\n}\n\n/**\n * Creates a distance matrix using a custom distance function.\n *\n * @export\n * @param {Vectors} data Input vectors to calculate distances.\n * @param {DistanceMetric} distance Custom distance function.\n * @return {Matrix} Calculated custom distance matrix.\n */\nexport function calcDistanceMatrix(data: Vectors, distance: DistanceMetric): Matrix {\n  const nItems = data.length;\n  const matrix = initCoordinates(nItems, nItems, 0);\n\n  for (let i = 0; i < nItems; ++i) {\n    for (let j = i + 1; j < nItems; ++j) {\n      const d: number = (data[i] == null) || (data[j] == null) ? 0 : distance(data[i], data[j]);\n      matrix[i][j] = matrix[j][i] = d;\n    }\n  }\n  return matrix;\n}\n\n/** Generates array from a range [begin; end] or [begin; end) if endExclusive. **/\nexport function genRange(begin: number, end: number, endExclusive = false): Int32Array {\n  const nItems = end - begin + (endExclusive ? 0 : 1);\n  const series = new Int32Array(nItems);\n\n  for (let i = 0; i < nItems; ++i)\n    series[i] = begin + i;\n\n  return series;\n}\n\n/**\n * Returns order of values as if they are sorted.\n *\n * @export\n * @param {any[]} values Input array.\n * @param {boolean} [reverse=false] Whether to return reversed order.\n * @return {number[]} The order computed.\n */\nexport function argSort(values: any[], reverse = false): number[] {\n  const sortfn = reverse ? (a: any[], b: any[]) => (b[0] - a[0]) : (a: any[], b: any[]) => (a[0] - b[0]);\n  const decor = (v: any, i: number) => [v, i]; // set index to value\n  const undecor = (a: any[]) => a[1]; // leave only index\n  const _argsort = (arr: any[]) => arr.map(decor).sort(sortfn).map(undecor);\n  return _argsort(values);\n}\n\n/**\n * Returns the indexes of the most diverse objects according to the dist function\n * @param {number} length total number of objects\n * @param {number} n number of diverse elements to find\n * @param {(i1: number, i2: number) => number} dist a function which calculates distance between\n *                                                  two objects using their indexes\n * @returns {number[]} The indexes of the most diverse objects\n */\nexport function getDiverseSubset(length: number, n: number, dist: (i1: number, i2: number) => number): number[] {\n  function maxBy(values: IterableIterator<number>, orderBy: (i: number) => number) {\n    let maxValue = null;\n    let maxOrderBy = null;\n\n    for (const element of values) {\n      const elementOrderBy = orderBy(element);\n      if (maxOrderBy == null || elementOrderBy > maxOrderBy) {\n        maxValue = element;\n        maxOrderBy = elementOrderBy;\n      }\n    }\n    return maxValue;\n  }\n\n  const subset = [randomInt(length - 1)];\n  const complement = new Set();\n\n  for (let i = 0; i < length; ++i) {\n    if (!subset.includes(i))\n      complement.add(i);\n  }\n\n  while (subset.length < n) {\n    const idx = maxBy(\n      complement.values() as IterableIterator<number>,\n      (i) => Math.min.apply(Math, subset.map(function(val, index) {\n        return dist(i, val);\n      })));\n    if (idx) {\n      subset.push(idx);\n      complement.delete(idx);\n    }\n  }\n  return subset;\n}\n\n/**\n * Returns normalized vector\n * @param {Vector} data numerical array\n */\nexport function normalize(data: Vector): Vector {\n  let mean = 0;\n  let std = 0;\n\n  for (let i = 0; i < data.length; ++i)\n    mean += data[i];\n\n  mean /= data.length;\n\n  for (let i = 0; i < data.length; ++i)\n    std += (data[i] - mean) * (data[i] - mean);\n\n  std = Math.sqrt(std / data.length);\n\n  for (let i = 0; i < data.length; ++i)\n    data[i] = (data[i] - mean) / std;\n\n  return data;\n}\n\n/**\n * Finds set difference between two lists.\n * @param {any[]} a The first list.\n * @param {any[]} b The second list.\n * @return {any[]}\n */\nexport function setDifference(a: any[], b: any[]): any[] {\n  const bSet = new Set(b);\n  return Array.from(new Set(a.filter((x) => !bSet.has(x))).values());\n}\n"]}","(function (root, factory) {\n    // Hack to make all exports of this module sha256 function object properties.\n    var exports = {};\n    factory(exports);\n    var sha256 = exports[\"default\"];\n    for (var k in exports) {\n        sha256[k] = exports[k];\n    }\n        \n    if (typeof module === 'object' && typeof module.exports === 'object') {\n        module.exports = sha256;\n    } else if (typeof define === 'function' && define.amd) {\n        define(function() { return sha256; }); \n    } else {\n        root.sha256 = sha256;\n    }\n})(this, function(exports) {\n\"use strict\";\nexports.__esModule = true;\n// SHA-256 (+ HMAC and PBKDF2) for JavaScript.\n//\n// Written in 2014-2016 by Dmitry Chestnykh.\n// Public domain, no warranty.\n//\n// Functions (accept and return Uint8Arrays):\n//\n//   sha256(message) -> hash\n//   sha256.hmac(key, message) -> mac\n//   sha256.pbkdf2(password, salt, rounds, dkLen) -> dk\n//\n//  Classes:\n//\n//   new sha256.Hash()\n//   new sha256.HMAC(key)\n//\nexports.digestLength = 32;\nexports.blockSize = 64;\n// SHA-256 constants\nvar K = new Uint32Array([\n    0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b,\n    0x59f111f1, 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01,\n    0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7,\n    0xc19bf174, 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc,\n    0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, 0x983e5152,\n    0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147,\n    0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc,\n    0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,\n    0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819,\n    0xd6990624, 0xf40e3585, 0x106aa070, 0x19a4c116, 0x1e376c08,\n    0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f,\n    0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208,\n    0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2\n]);\nfunction hashBlocks(w, v, p, pos, len) {\n    var a, b, c, d, e, f, g, h, u, i, j, t1, t2;\n    while (len >= 64) {\n        a = v[0];\n        b = v[1];\n        c = v[2];\n        d = v[3];\n        e = v[4];\n        f = v[5];\n        g = v[6];\n        h = v[7];\n        for (i = 0; i < 16; i++) {\n            j = pos + i * 4;\n            w[i] = (((p[j] & 0xff) << 24) | ((p[j + 1] & 0xff) << 16) |\n                ((p[j + 2] & 0xff) << 8) | (p[j + 3] & 0xff));\n        }\n        for (i = 16; i < 64; i++) {\n            u = w[i - 2];\n            t1 = (u >>> 17 | u << (32 - 17)) ^ (u >>> 19 | u << (32 - 19)) ^ (u >>> 10);\n            u = w[i - 15];\n            t2 = (u >>> 7 | u << (32 - 7)) ^ (u >>> 18 | u << (32 - 18)) ^ (u >>> 3);\n            w[i] = (t1 + w[i - 7] | 0) + (t2 + w[i - 16] | 0);\n        }\n        for (i = 0; i < 64; i++) {\n            t1 = (((((e >>> 6 | e << (32 - 6)) ^ (e >>> 11 | e << (32 - 11)) ^\n                (e >>> 25 | e << (32 - 25))) + ((e & f) ^ (~e & g))) | 0) +\n                ((h + ((K[i] + w[i]) | 0)) | 0)) | 0;\n            t2 = (((a >>> 2 | a << (32 - 2)) ^ (a >>> 13 | a << (32 - 13)) ^\n                (a >>> 22 | a << (32 - 22))) + ((a & b) ^ (a & c) ^ (b & c))) | 0;\n            h = g;\n            g = f;\n            f = e;\n            e = (d + t1) | 0;\n            d = c;\n            c = b;\n            b = a;\n            a = (t1 + t2) | 0;\n        }\n        v[0] += a;\n        v[1] += b;\n        v[2] += c;\n        v[3] += d;\n        v[4] += e;\n        v[5] += f;\n        v[6] += g;\n        v[7] += h;\n        pos += 64;\n        len -= 64;\n    }\n    return pos;\n}\n// Hash implements SHA256 hash algorithm.\nvar Hash = /** @class */ (function () {\n    function Hash() {\n        this.digestLength = exports.digestLength;\n        this.blockSize = exports.blockSize;\n        // Note: Int32Array is used instead of Uint32Array for performance reasons.\n        this.state = new Int32Array(8); // hash state\n        this.temp = new Int32Array(64); // temporary state\n        this.buffer = new Uint8Array(128); // buffer for data to hash\n        this.bufferLength = 0; // number of bytes in buffer\n        this.bytesHashed = 0; // number of total bytes hashed\n        this.finished = false; // indicates whether the hash was finalized\n        this.reset();\n    }\n    // Resets hash state making it possible\n    // to re-use this instance to hash other data.\n    Hash.prototype.reset = function () {\n        this.state[0] = 0x6a09e667;\n        this.state[1] = 0xbb67ae85;\n        this.state[2] = 0x3c6ef372;\n        this.state[3] = 0xa54ff53a;\n        this.state[4] = 0x510e527f;\n        this.state[5] = 0x9b05688c;\n        this.state[6] = 0x1f83d9ab;\n        this.state[7] = 0x5be0cd19;\n        this.bufferLength = 0;\n        this.bytesHashed = 0;\n        this.finished = false;\n        return this;\n    };\n    // Cleans internal buffers and re-initializes hash state.\n    Hash.prototype.clean = function () {\n        for (var i = 0; i < this.buffer.length; i++) {\n            this.buffer[i] = 0;\n        }\n        for (var i = 0; i < this.temp.length; i++) {\n            this.temp[i] = 0;\n        }\n        this.reset();\n    };\n    // Updates hash state with the given data.\n    //\n    // Optionally, length of the data can be specified to hash\n    // fewer bytes than data.length.\n    //\n    // Throws error when trying to update already finalized hash:\n    // instance must be reset to use it again.\n    Hash.prototype.update = function (data, dataLength) {\n        if (dataLength === void 0) { dataLength = data.length; }\n        if (this.finished) {\n            throw new Error(\"SHA256: can't update because hash was finished.\");\n        }\n        var dataPos = 0;\n        this.bytesHashed += dataLength;\n        if (this.bufferLength > 0) {\n            while (this.bufferLength < 64 && dataLength > 0) {\n                this.buffer[this.bufferLength++] = data[dataPos++];\n                dataLength--;\n            }\n            if (this.bufferLength === 64) {\n                hashBlocks(this.temp, this.state, this.buffer, 0, 64);\n                this.bufferLength = 0;\n            }\n        }\n        if (dataLength >= 64) {\n            dataPos = hashBlocks(this.temp, this.state, data, dataPos, dataLength);\n            dataLength %= 64;\n        }\n        while (dataLength > 0) {\n            this.buffer[this.bufferLength++] = data[dataPos++];\n            dataLength--;\n        }\n        return this;\n    };\n    // Finalizes hash state and puts hash into out.\n    //\n    // If hash was already finalized, puts the same value.\n    Hash.prototype.finish = function (out) {\n        if (!this.finished) {\n            var bytesHashed = this.bytesHashed;\n            var left = this.bufferLength;\n            var bitLenHi = (bytesHashed / 0x20000000) | 0;\n            var bitLenLo = bytesHashed << 3;\n            var padLength = (bytesHashed % 64 < 56) ? 64 : 128;\n            this.buffer[left] = 0x80;\n            for (var i = left + 1; i < padLength - 8; i++) {\n                this.buffer[i] = 0;\n            }\n            this.buffer[padLength - 8] = (bitLenHi >>> 24) & 0xff;\n            this.buffer[padLength - 7] = (bitLenHi >>> 16) & 0xff;\n            this.buffer[padLength - 6] = (bitLenHi >>> 8) & 0xff;\n            this.buffer[padLength - 5] = (bitLenHi >>> 0) & 0xff;\n            this.buffer[padLength - 4] = (bitLenLo >>> 24) & 0xff;\n            this.buffer[padLength - 3] = (bitLenLo >>> 16) & 0xff;\n            this.buffer[padLength - 2] = (bitLenLo >>> 8) & 0xff;\n            this.buffer[padLength - 1] = (bitLenLo >>> 0) & 0xff;\n            hashBlocks(this.temp, this.state, this.buffer, 0, padLength);\n            this.finished = true;\n        }\n        for (var i = 0; i < 8; i++) {\n            out[i * 4 + 0] = (this.state[i] >>> 24) & 0xff;\n            out[i * 4 + 1] = (this.state[i] >>> 16) & 0xff;\n            out[i * 4 + 2] = (this.state[i] >>> 8) & 0xff;\n            out[i * 4 + 3] = (this.state[i] >>> 0) & 0xff;\n        }\n        return this;\n    };\n    // Returns the final hash digest.\n    Hash.prototype.digest = function () {\n        var out = new Uint8Array(this.digestLength);\n        this.finish(out);\n        return out;\n    };\n    // Internal function for use in HMAC for optimization.\n    Hash.prototype._saveState = function (out) {\n        for (var i = 0; i < this.state.length; i++) {\n            out[i] = this.state[i];\n        }\n    };\n    // Internal function for use in HMAC for optimization.\n    Hash.prototype._restoreState = function (from, bytesHashed) {\n        for (var i = 0; i < this.state.length; i++) {\n            this.state[i] = from[i];\n        }\n        this.bytesHashed = bytesHashed;\n        this.finished = false;\n        this.bufferLength = 0;\n    };\n    return Hash;\n}());\nexports.Hash = Hash;\n// HMAC implements HMAC-SHA256 message authentication algorithm.\nvar HMAC = /** @class */ (function () {\n    function HMAC(key) {\n        this.inner = new Hash();\n        this.outer = new Hash();\n        this.blockSize = this.inner.blockSize;\n        this.digestLength = this.inner.digestLength;\n        var pad = new Uint8Array(this.blockSize);\n        if (key.length > this.blockSize) {\n            (new Hash()).update(key).finish(pad).clean();\n        }\n        else {\n            for (var i = 0; i < key.length; i++) {\n                pad[i] = key[i];\n            }\n        }\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] ^= 0x36;\n        }\n        this.inner.update(pad);\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] ^= 0x36 ^ 0x5c;\n        }\n        this.outer.update(pad);\n        this.istate = new Uint32Array(8);\n        this.ostate = new Uint32Array(8);\n        this.inner._saveState(this.istate);\n        this.outer._saveState(this.ostate);\n        for (var i = 0; i < pad.length; i++) {\n            pad[i] = 0;\n        }\n    }\n    // Returns HMAC state to the state initialized with key\n    // to make it possible to run HMAC over the other data with the same\n    // key without creating a new instance.\n    HMAC.prototype.reset = function () {\n        this.inner._restoreState(this.istate, this.inner.blockSize);\n        this.outer._restoreState(this.ostate, this.outer.blockSize);\n        return this;\n    };\n    // Cleans HMAC state.\n    HMAC.prototype.clean = function () {\n        for (var i = 0; i < this.istate.length; i++) {\n            this.ostate[i] = this.istate[i] = 0;\n        }\n        this.inner.clean();\n        this.outer.clean();\n    };\n    // Updates state with provided data.\n    HMAC.prototype.update = function (data) {\n        this.inner.update(data);\n        return this;\n    };\n    // Finalizes HMAC and puts the result in out.\n    HMAC.prototype.finish = function (out) {\n        if (this.outer.finished) {\n            this.outer.finish(out);\n        }\n        else {\n            this.inner.finish(out);\n            this.outer.update(out, this.digestLength).finish(out);\n        }\n        return this;\n    };\n    // Returns message authentication code.\n    HMAC.prototype.digest = function () {\n        var out = new Uint8Array(this.digestLength);\n        this.finish(out);\n        return out;\n    };\n    return HMAC;\n}());\nexports.HMAC = HMAC;\n// Returns SHA256 hash of data.\nfunction hash(data) {\n    var h = (new Hash()).update(data);\n    var digest = h.digest();\n    h.clean();\n    return digest;\n}\nexports.hash = hash;\n// Function hash is both available as module.hash and as default export.\nexports[\"default\"] = hash;\n// Returns HMAC-SHA256 of data under the key.\nfunction hmac(key, data) {\n    var h = (new HMAC(key)).update(data);\n    var digest = h.digest();\n    h.clean();\n    return digest;\n}\nexports.hmac = hmac;\n// Fills hkdf buffer like this:\n// T(1) = HMAC-Hash(PRK, T(0) | info | 0x01)\nfunction fillBuffer(buffer, hmac, info, counter) {\n    // Counter is a byte value: check if it overflowed.\n    var num = counter[0];\n    if (num === 0) {\n        throw new Error(\"hkdf: cannot expand more\");\n    }\n    // Prepare HMAC instance for new data with old key.\n    hmac.reset();\n    // Hash in previous output if it was generated\n    // (i.e. counter is greater than 1).\n    if (num > 1) {\n        hmac.update(buffer);\n    }\n    // Hash in info if it exists.\n    if (info) {\n        hmac.update(info);\n    }\n    // Hash in the counter.\n    hmac.update(counter);\n    // Output result to buffer and clean HMAC instance.\n    hmac.finish(buffer);\n    // Increment counter inside typed array, this works properly.\n    counter[0]++;\n}\nvar hkdfSalt = new Uint8Array(exports.digestLength); // Filled with zeroes.\nfunction hkdf(key, salt, info, length) {\n    if (salt === void 0) { salt = hkdfSalt; }\n    if (length === void 0) { length = 32; }\n    var counter = new Uint8Array([1]);\n    // HKDF-Extract uses salt as HMAC key, and key as data.\n    var okm = hmac(salt, key);\n    // Initialize HMAC for expanding with extracted key.\n    // Ensure no collisions with `hmac` function.\n    var hmac_ = new HMAC(okm);\n    // Allocate buffer.\n    var buffer = new Uint8Array(hmac_.digestLength);\n    var bufpos = buffer.length;\n    var out = new Uint8Array(length);\n    for (var i = 0; i < length; i++) {\n        if (bufpos === buffer.length) {\n            fillBuffer(buffer, hmac_, info, counter);\n            bufpos = 0;\n        }\n        out[i] = buffer[bufpos++];\n    }\n    hmac_.clean();\n    buffer.fill(0);\n    counter.fill(0);\n    return out;\n}\nexports.hkdf = hkdf;\n// Derives a key from password and salt using PBKDF2-HMAC-SHA256\n// with the given number of iterations.\n//\n// The number of bytes returned is equal to dkLen.\n//\n// (For better security, avoid dkLen greater than hash length - 32 bytes).\nfunction pbkdf2(password, salt, iterations, dkLen) {\n    var prf = new HMAC(password);\n    var len = prf.digestLength;\n    var ctr = new Uint8Array(4);\n    var t = new Uint8Array(len);\n    var u = new Uint8Array(len);\n    var dk = new Uint8Array(dkLen);\n    for (var i = 0; i * len < dkLen; i++) {\n        var c = i + 1;\n        ctr[0] = (c >>> 24) & 0xff;\n        ctr[1] = (c >>> 16) & 0xff;\n        ctr[2] = (c >>> 8) & 0xff;\n        ctr[3] = (c >>> 0) & 0xff;\n        prf.reset();\n        prf.update(salt);\n        prf.update(ctr);\n        prf.finish(u);\n        for (var j = 0; j < len; j++) {\n            t[j] = u[j];\n        }\n        for (var j = 2; j <= iterations; j++) {\n            prf.reset();\n            prf.update(u).finish(u);\n            for (var k = 0; k < len; k++) {\n                t[k] ^= u[k];\n            }\n        }\n        for (var j = 0; j < len && i * len + j < dkLen; j++) {\n            dk[i * len + j] = t[j];\n        }\n    }\n    for (var i = 0; i < len; i++) {\n        t[i] = u[i] = 0;\n    }\n    for (var i = 0; i < 4; i++) {\n        ctr[i] = 0;\n    }\n    prf.clean();\n    return dk;\n}\nexports.pbkdf2 = pbkdf2;\n});\n","\"use strict\";\nconst peq = new Uint32Array(0x10000);\nconst myers_32 = (a, b) => {\n  const n = a.length;\n  const m = b.length;\n  const lst = 1 << (n - 1);\n  let pv = -1;\n  let mv = 0;\n  let sc = n;\n  let i = n;\n  while (i--) {\n    peq[a.charCodeAt(i)] |= 1 << i;\n  }\n  for (i = 0; i < m; i++) {\n    let eq = peq[b.charCodeAt(i)];\n    const xv = eq | mv;\n    eq |= ((eq & pv) + pv) ^ pv;\n    mv |= ~(eq | pv);\n    pv &= eq;\n    if (mv & lst) {\n      sc++;\n    }\n    if (pv & lst) {\n      sc--;\n    }\n    mv = (mv << 1) | 1;\n    pv = (pv << 1) | ~(xv | mv);\n    mv &= xv;\n  }\n  i = n;\n  while (i--) {\n    peq[a.charCodeAt(i)] = 0;\n  }\n  return sc;\n};\n\nconst myers_x = (a, b) => {\n  const n = a.length;\n  const m = b.length;\n  const mhc = [];\n  const phc = [];\n  const hsize = Math.ceil(n / 32);\n  const vsize = Math.ceil(m / 32);\n  let score = m;\n  for (let i = 0; i < hsize; i++) {\n    phc[i] = -1;\n    mhc[i] = 0;\n  }\n  let j = 0;\n  for (; j < vsize - 1; j++) {\n    let mv = 0;\n    let pv = -1;\n    const start = j * 32;\n    const end = Math.min(32, m) + start;\n    for (let k = start; k < end; k++) {\n      peq[b.charCodeAt(k)] |= 1 << k;\n    }\n    score = m;\n    for (let i = 0; i < n; i++) {\n      const eq = peq[a.charCodeAt(i)];\n      const pb = (phc[(i / 32) | 0] >>> i) & 1;\n      const mb = (mhc[(i / 32) | 0] >>> i) & 1;\n      const xv = eq | mv;\n      const xh = ((((eq | mb) & pv) + pv) ^ pv) | eq | mb;\n      let ph = mv | ~(xh | pv);\n      let mh = pv & xh;\n      if ((ph >>> 31) ^ pb) {\n        phc[(i / 32) | 0] ^= 1 << i;\n      }\n      if ((mh >>> 31) ^ mb) {\n        mhc[(i / 32) | 0] ^= 1 << i;\n      }\n      ph = (ph << 1) | pb;\n      mh = (mh << 1) | mb;\n      pv = mh | ~(xv | ph);\n      mv = ph & xv;\n    }\n    for (let k = start; k < end; k++) {\n      peq[b.charCodeAt(k)] = 0;\n    }\n  }\n  let mv = 0;\n  let pv = -1;\n  const start = j * 32;\n  const end = Math.min(32, m - start) + start;\n  for (let k = start; k < end; k++) {\n    peq[b.charCodeAt(k)] |= 1 << k;\n  }\n  score = m;\n  for (let i = 0; i < n; i++) {\n    const eq = peq[a.charCodeAt(i)];\n    const pb = (phc[(i / 32) | 0] >>> i) & 1;\n    const mb = (mhc[(i / 32) | 0] >>> i) & 1;\n    const xv = eq | mv;\n    const xh = ((((eq | mb) & pv) + pv) ^ pv) | eq | mb;\n    let ph = mv | ~(xh | pv);\n    let mh = pv & xh;\n    score += (ph >>> (m - 1)) & 1;\n    score -= (mh >>> (m - 1)) & 1;\n    if ((ph >>> 31) ^ pb) {\n      phc[(i / 32) | 0] ^= 1 << i;\n    }\n    if ((mh >>> 31) ^ mb) {\n      mhc[(i / 32) | 0] ^= 1 << i;\n    }\n    ph = (ph << 1) | pb;\n    mh = (mh << 1) | mb;\n    pv = mh | ~(xv | ph);\n    mv = ph & xv;\n  }\n  for (let k = start; k < end; k++) {\n    peq[b.charCodeAt(k)] = 0;\n  }\n  return score;\n};\n\nconst distance = (a, b) => {\n  if (a.length > b.length) {\n    const tmp = b;\n    b = a;\n    a = tmp;\n  }\n  if (a.length === 0) {\n    return b.length;\n  }\n  if (a.length <= 32) {\n    return myers_32(a, b);\n  }\n  return myers_x(a, b);\n};\n\nconst closest = (str, arr) => {\n  let min_distance = Infinity;\n  let min_index = 0;\n  for (let i = 0; i < arr.length; i++) {\n    const dist = distance(str, arr[i]);\n    if (dist < min_distance) {\n      min_distance = dist;\n      min_index = i;\n    }\n  }\n  return arr[min_index];\n};\n\nmodule.exports = {\n  closest, distance\n}\n","\"use strict\";\n// Reference: https://www.geeksforgeeks.org/jaro-and-jaro-winkler-similarity/\nObject.defineProperty(exports, \"__esModule\", { value: true });\nexports.jaroWinkler = exports.jaro = void 0;\n/**\n *\n * @param str1 String 1 for compare\n * @param str2 String 2 for compare\n * @param options to control case sensitive or not\n */\nfunction jaro(str1, str2, options) {\n    // Exit early if either are empty.\n    if (str1.length === 0 || str2.length === 0) {\n        return 0;\n    }\n    // Convert to upper if case-sensitive is false.\n    if (options && !options.caseSensitive) {\n        str1 = str1.toUpperCase();\n        str2 = str2.toUpperCase();\n    }\n    // Exact match\n    if (str1 === str2) {\n        return 1;\n    }\n    // Number of matches\n    var m = 0;\n    // Length of two Strings\n    var len1 = str1.length;\n    var len2 = str2.length;\n    // Maximum distance\n    var window = Math.floor(Math.max(len1, len2) / 2) - 1;\n    // Hash for matches\n    var str1Hash = new Array(len1);\n    var str2Hash = new Array(len2);\n    for (var i = 0; i < len1; i++) {\n        for (var j = Math.max(0, i - window); j <= Math.min(len2, i + window + 1); j++) {\n            if (!str1Hash[i] && !str2Hash[j] && str1[i] === str2[j]) {\n                ++m;\n                str1Hash[i] = str2Hash[j] = true;\n                break;\n            }\n        }\n    }\n    // Exit early if no matches were found.\n    if (m === 0) {\n        return 0;\n    }\n    // Count the transpositions.\n    var t = 0;\n    var point = 0;\n    for (var i = 0; i < len1; i++) {\n        if (str1Hash[i]) {\n            while (!str2Hash[point]) {\n                point++;\n            }\n            if (str1.charAt(i) !== str2.charAt(point++)) {\n                t++;\n            }\n        }\n    }\n    t /= 2;\n    return (m / len1 + m / len2 + (m - t) / m) / 3;\n}\nexports.jaro = jaro;\n/**\n *\n * @param str1 String 1 for compare\n * @param str2 String 2 for compare\n * @param options to control case sensitive or not\n */\nfunction jaroWinkler(str1, str2, options) {\n    // Jaro Distance\n    var jaroDist = jaro(str1, str2, options);\n    // Same prefix length, maxium is 4\n    var prefix = 0;\n    if (jaroDist > 0.7) {\n        var minIndex = Math.min(str1.length, str2.length);\n        var i = 0;\n        while (str1[i] === str2[i] && i < 4 && i < minIndex) {\n            ++prefix;\n            i++;\n        }\n        jaroDist += 0.1 * prefix * (1 - jaroDist);\n    }\n    return jaroDist;\n}\nexports.jaroWinkler = jaroWinkler;\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nconst V2000_ATOM_NAME_POS = 31;\nexport function getFingerprints(mols, monomers) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const mod = yield grok.functions.call('Chem:getRdKitModule');\n        const fps = [];\n        let dict = new Map();\n        for (let i = 0; i < monomers.length; i++)\n            dict.set(monomers[i], `R${Math.pow(10, (i + 1))}`);\n        mols = changeToV3000(mols, dict);\n        for (let i = 0; i < mols.length; i++) {\n            const mol = mod.get_mol(mols[i]);\n            const fp = mol.get_pattern_fp_as_uint8array();\n            fps.push(fp);\n            mol === null || mol === void 0 ? void 0 : mol.delete();\n        }\n        return fps;\n    });\n}\nfunction changeToV3000(mols, dict) {\n    for (let i = 0; i < mols.length; i++) {\n        let curPos = 0;\n        let endPos = 0;\n        let molV3000 = `\n    Datagrok macromolecule handler\n\n  0  0  0  0  0  0            999 V3000\nM  V30 BEGIN CTAB\n`;\n        const mol = mols[i];\n        curPos = mol.indexOf('\\n', curPos) + 1;\n        curPos = mol.indexOf('\\n', curPos) + 1;\n        curPos = mol.indexOf('\\n', curPos) + 1;\n        const atomMonomerCounts = parseInt(mol.substring(curPos, curPos + 3));\n        const bondMonomerCounts = parseInt(mol.substring(curPos + 3, curPos + 6));\n        molV3000 += `M  V30 COUNTS ${atomMonomerCounts} ${bondMonomerCounts} 0 0 0\\n`;\n        molV3000 += 'M  V30 BEGIN ATOM\\n';\n        for (let atomRowI = 0; atomRowI < atomMonomerCounts; atomRowI++) {\n            curPos = mol.indexOf('\\n', curPos) + 1 + V2000_ATOM_NAME_POS;\n            endPos = mol.indexOf(' ', curPos);\n            const monomerName = mol.substring(curPos, endPos);\n            molV3000 += `M  V30 ${atomRowI + 1} ${dict.get(monomerName)} 0.000 0.000 0 0\\n`;\n        }\n        molV3000 += 'M  V30 END ATOM\\n';\n        molV3000 += 'M  V30 BEGIN BOND\\n';\n        for (let bondRowI = 0; bondRowI < bondMonomerCounts; bondRowI++) {\n            curPos = mol.indexOf('\\n', curPos) + 1;\n            const firstMonomer = parseInt(mol.substring(curPos, curPos + 3).trim());\n            const secondMonomer = parseInt(mol.substring(curPos + 3, curPos + 6).trim());\n            const order = parseInt(mol.substring(curPos + 6, curPos + 9).trim());\n            molV3000 += `M  V30 ${bondRowI + 1} ${order} ${firstMonomer} ${secondMonomer}\\n`;\n        }\n        molV3000 += 'M  V30 END BOND\\n';\n        molV3000 += 'M  V30 END CTAB\\n';\n        molV3000 += 'M  END';\n        console.log(molV3000);\n        mols[i] = molV3000;\n    }\n    return mols;\n}\n","export const jsonSdfMonomerLibDict = {\n    'monomerType': null,\n    'smiles': null,\n    'name': 'MonomerName',\n    'author': null,\n    'molfile': 'molecule',\n    'naturalAnalog': 'MonomerNaturalAnalogCode',\n    'rgroups': 'MonomerCaps',\n    'createDate': null,\n    'id': null,\n    'polymerType': 'MonomerType',\n    'symbol': 'MonomerCode'\n};\nexport const RGROUP_FIELD = 'rgroups';\nexport const CAP_GROUP_SMILES = 'capGroupSmiles';\nexport const RGROUP_ALTER_ID = 'alternateId';\nexport const CAP_GROUP_NAME = 'capGroupName';\nexport const RGROUP_LABEL = 'label';\nexport const MONOMER_SYMBOL = 'symbol';\nexport const SDF_MONOMER_NAME = 'MonomerName';\n// range of hex nubers used in PepSea library to endode monomers\nexport const MONOMER_ENCODE_MIN = 0x100;\nexport const MONOMER_ENCODE_MAX = 0x40A;\n","class Cell {\n    constructor() {\n        this.value = 0;\n        this.parentI = 0;\n        this.parentJ = 0;\n    }\n}\n;\nexport class Aligned {\n    constructor(seq1_, seq2_, score_) {\n        this.seq1 = seq1_;\n        this.seq2 = seq2_;\n        this.score = score_;\n    }\n}\n;\nexport class SequenceAlignment {\n    constructor(seq1_, seq2_, gap_, alignGrid_ = \"\") {\n        this.seq1 = '';\n        this.seq2 = '';\n        this.gap = 1;\n        this.len1 = 0;\n        this.len2 = 0;\n        this.step = 1;\n        this.alignGrid = [];\n        this.scores = [];\n        this.dpGrid = [];\n        this.connections = {};\n        this.BLOSUM45 = [[5, -2, -1, -2, -1, -1, -1, 0, -2, -1, -1, -1, -1, -2, -1, 1, 0, -2, -2, 0, -1, -1, 0, -5],\n            [-2, 7, 0, -1, -3, 1, 0, -2, 0, -3, -2, 3, -1, -2, -2, -1, -1, -2, -1, -2, -1, 0, -1, -5],\n            [-1, 0, 6, 2, -2, 0, 0, 0, 1, -2, -3, 0, -2, -2, -2, 1, 0, -4, -2, -3, 4, 0, -1, -5],\n            [-2, -1, 2, 7, -3, 0, 2, -1, 0, -4, -3, 0, -3, -4, -1, 0, -1, -4, -2, -3, 5, 1, -1, -5],\n            [-1, -3, -2, -3, 12, -3, -3, -3, -3, -3, -2, -3, -2, -2, -4, -1, -1, -5, -3, -1, -2, -3, -2, -5],\n            [-1, 1, 0, 0, -3, 6, 2, -2, 1, -2, -2, 1, 0, -4, -1, 0, -1, -2, -1, -3, 0, 4, -1, -5],\n            [-1, 0, 0, 2, -3, 2, 6, -2, 0, -3, -2, 1, -2, -3, 0, 0, -1, -3, -2, -3, 1, 4, -1, -5],\n            [0, -2, 0, -1, -3, -2, -2, 7, -2, -4, -3, -2, -2, -3, -2, 0, -2, -2, -3, -3, -1, -2, -1, -5],\n            [-2, 0, 1, 0, -3, 1, 0, -2, 10, -3, -2, -1, 0, -2, -2, -1, -2, -3, 2, -3, 0, 0, -1, -5],\n            [-1, -3, -2, -4, -3, -2, -3, -4, -3, 5, 2, -3, 2, 0, -2, -2, -1, -2, 0, 3, -3, -3, -1, -5],\n            [-1, -2, -3, -3, -2, -2, -2, -3, -2, 2, 5, -3, 2, 1, -3, -3, -1, -2, 0, 1, -3, -2, -1, -5],\n            [-1, 3, 0, 0, -3, 1, 1, -2, -1, -3, -3, 5, -1, -3, -1, -1, -1, -2, -1, -2, 0, 1, -1, -5],\n            [-1, -1, -2, -3, -2, 0, -2, -2, 0, 2, 2, -1, 6, 0, -2, -2, -1, -2, 0, 1, -2, -1, -1, -5],\n            [-2, -2, -2, -4, -2, -4, -3, -3, -2, 0, 1, -3, 0, 8, -3, -2, -1, 1, 3, 0, -3, -3, -1, -5],\n            [-1, -2, -2, -1, -4, -1, 0, -2, -2, -2, -3, -1, -2, -3, 9, -1, -1, -3, -3, -3, -2, -1, -1, -5],\n            [1, -1, 1, 0, -1, 0, 0, 0, -1, -2, -3, -1, -2, -2, -1, 4, 2, -4, -2, -1, 0, 0, 0, -5],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -1, -1, 2, 5, -3, -1, 0, 0, -1, 0, -5],\n            [-2, -2, -4, -4, -5, -2, -3, -2, -3, -2, -2, -2, -2, 1, -3, -4, -3, 15, 3, -3, -4, -2, -2, -5],\n            [-2, -1, -2, -2, -3, -1, -2, -3, 2, 0, 0, -1, 0, 3, -3, -2, -1, 3, 8, -1, -2, -2, -1, -5],\n            [0, -2, -3, -3, -1, -3, -3, -3, -3, 3, 1, -2, 1, 0, -3, -1, 0, -3, -1, 5, -3, -3, -1, -5],\n            [-1, -1, 4, 5, -2, 0, 1, -1, 0, -3, -3, 0, -2, -3, -2, 0, 0, -4, -2, -3, 4, 2, -1, -5],\n            [-1, 0, 0, 1, -3, 4, 4, -2, 0, -3, -2, 1, -1, -3, -1, 0, -1, -2, -2, -3, 2, 4, -1, -5],\n            [0, -1, -1, -1, -2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 0, -2, -1, -1, -1, -1, -1, -5],\n            [-5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, 1]];\n        this.BLOSUM50 = [[5, -2, -1, -2, -1, -1, -1, 0, -2, -1, -2, -1, -1, -3, -1, 1, 0, -3, -2, 0, -2, -1, -1, -5],\n            [-2, 7, -1, -2, -4, 1, 0, -3, 0, -4, -3, 3, -2, -3, -3, -1, -1, -3, -1, -3, -1, 0, -1, -5],\n            [-1, -1, 7, 2, -2, 0, 0, 0, 1, -3, -4, 0, -2, -4, -2, 1, 0, -4, -2, -3, 4, 0, -1, -5],\n            [-2, -2, 2, 8, -4, 0, 2, -1, -1, -4, -4, -1, -4, -5, -1, 0, -1, -5, -3, -4, 5, 1, -1, -5],\n            [-1, -4, -2, -4, 13, -3, -3, -3, -3, -2, -2, -3, -2, -2, -4, -1, -1, -5, -3, -1, -3, -3, -2, -5],\n            [-1, 1, 0, 0, -3, 7, 2, -2, 1, -3, -2, 2, 0, -4, -1, 0, -1, -1, -1, -3, 0, 4, -1, -5],\n            [-1, 0, 0, 2, -3, 2, 6, -3, 0, -4, -3, 1, -2, -3, -1, -1, -1, -3, -2, -3, 1, 5, -1, -5],\n            [0, -3, 0, -1, -3, -2, -3, 8, -2, -4, -4, -2, -3, -4, -2, 0, -2, -3, -3, -4, -1, -2, -2, -5],\n            [-2, 0, 1, -1, -3, 1, 0, -2, 10, -4, -3, 0, -1, -1, -2, -1, -2, -3, 2, -4, 0, 0, -1, -5],\n            [-1, -4, -3, -4, -2, -3, -4, -4, -4, 5, 2, -3, 2, 0, -3, -3, -1, -3, -1, 4, -4, -3, -1, -5],\n            [-2, -3, -4, -4, -2, -2, -3, -4, -3, 2, 5, -3, 3, 1, -4, -3, -1, -2, -1, 1, -4, -3, -1, -5],\n            [-1, 3, 0, -1, -3, 2, 1, -2, 0, -3, -3, 6, -2, -4, -1, 0, -1, -3, -2, -3, 0, 1, -1, -5],\n            [-1, -2, -2, -4, -2, 0, -2, -3, -1, 2, 3, -2, 7, 0, -3, -2, -1, -1, 0, 1, -3, -1, -1, -5],\n            [-3, -3, -4, -5, -2, -4, -3, -4, -1, 0, 1, -4, 0, 8, -4, -3, -2, 1, 4, -1, -4, -4, -2, -5],\n            [-1, -3, -2, -1, -4, -1, -1, -2, -2, -3, -4, -1, -3, -4, 10, -1, -1, -4, -3, -3, -2, -1, -2, -5],\n            [1, -1, 1, 0, -1, 0, -1, 0, -1, -3, -3, 0, -2, -3, -1, 5, 2, -4, -2, -2, 0, 0, -1, -5],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -2, -1, 2, 5, -3, -2, 0, 0, -1, 0, -5],\n            [-3, -3, -4, -5, -5, -1, -3, -3, -3, -3, -2, -3, -1, 1, -4, -4, -3, 15, 2, -3, -5, -2, -3, -5],\n            [-2, -1, -2, -3, -3, -1, -2, -3, 2, -1, -1, -2, 0, 4, -3, -2, -2, 2, 8, -1, -3, -2, -1, -5],\n            [0, -3, -3, -4, -1, -3, -3, -4, -4, 4, 1, -3, 1, -1, -3, -2, 0, -3, -1, 5, -4, -3, -1, -5],\n            [-2, -1, 4, 5, -3, 0, 1, -1, 0, -4, -4, 0, -3, -4, -2, 0, 0, -5, -3, -4, 5, 2, -1, -5],\n            [-1, 0, 0, 1, -3, 4, 5, -2, 0, -3, -3, 1, -1, -4, -1, 0, -1, -2, -2, -3, 2, 5, -1, -5],\n            [-1, -1, -1, -1, -2, -1, -1, -2, -1, -1, -1, -1, -1, -2, -2, -1, 0, -3, -1, -1, -1, -1, -1, -5],\n            [-5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, -5, 1]];\n        this.BLOSUM62 = [[4, -1, -2, -2, 0, -1, -1, 0, -2, -1, -1, -1, -1, -2, -1, 1, 0, -3, -2, 0, -2, -1, 0, -4],\n            [-1, 5, 0, -2, -3, 1, 0, -2, 0, -3, -2, 2, -1, -3, -2, -1, -1, -3, -2, -3, -1, 0, -1, -4],\n            [-2, 0, 6, 1, -3, 0, 0, 0, 1, -3, -3, 0, -2, -3, -2, 1, 0, -4, -2, -3, 3, 0, -1, -4],\n            [-2, -2, 1, 6, -3, 0, 2, -1, -1, -3, -4, -1, -3, -3, -1, 0, -1, -4, -3, -3, 4, 1, -1, -4],\n            [0, -3, -3, -3, 9, -3, -4, -3, -3, -1, -1, -3, -1, -2, -3, -1, -1, -2, -2, -1, -3, -3, -2, -4],\n            [-1, 1, 0, 0, -3, 5, 2, -2, 0, -3, -2, 1, 0, -3, -1, 0, -1, -2, -1, -2, 0, 3, -1, -4],\n            [-1, 0, 0, 2, -4, 2, 5, -2, 0, -3, -3, 1, -2, -3, -1, 0, -1, -3, -2, -2, 1, 4, -1, -4],\n            [0, -2, 0, -1, -3, -2, -2, 6, -2, -4, -4, -2, -3, -3, -2, 0, -2, -2, -3, -3, -1, -2, -1, -4],\n            [-2, 0, 1, -1, -3, 0, 0, -2, 8, -3, -3, -1, -2, -1, -2, -1, -2, -2, 2, -3, 0, 0, -1, -4],\n            [-1, -3, -3, -3, -1, -3, -3, -4, -3, 4, 2, -3, 1, 0, -3, -2, -1, -3, -1, 3, -3, -3, -1, -4],\n            [-1, -2, -3, -4, -1, -2, -3, -4, -3, 2, 4, -2, 2, 0, -3, -2, -1, -2, -1, 1, -4, -3, -1, -4],\n            [-1, 2, 0, -1, -3, 1, 1, -2, -1, -3, -2, 5, -1, -3, -1, 0, -1, -3, -2, -2, 0, 1, -1, -4],\n            [-1, -1, -2, -3, -1, 0, -2, -3, -2, 1, 2, -1, 5, 0, -2, -1, -1, -1, -1, 1, -3, -1, -1, -4],\n            [-2, -3, -3, -3, -2, -3, -3, -3, -1, 0, 0, -3, 0, 6, -4, -2, -2, 1, 3, -1, -3, -3, -1, -4],\n            [-1, -2, -2, -1, -3, -1, -1, -2, -2, -3, -3, -1, -2, -4, 7, -1, -1, -4, -3, -2, -2, -1, -2, -4],\n            [1, -1, 1, 0, -1, 0, 0, 0, -1, -2, -2, 0, -1, -2, -1, 4, 1, -3, -2, -2, 0, 0, 0, -4],\n            [0, -1, 0, -1, -1, -1, -1, -2, -2, -1, -1, -1, -1, -2, -1, 1, 5, -2, -2, 0, -1, -1, 0, -4],\n            [-3, -3, -4, -4, -2, -2, -3, -2, -2, -3, -2, -3, -1, 1, -4, -3, -2, 11, 2, -3, -4, -3, -2, -4],\n            [-2, -2, -2, -3, -2, -1, -2, -3, 2, -1, -1, -2, -1, 3, -3, -2, -2, 2, 7, -1, -3, -2, -1, -4],\n            [0, -3, -3, -3, -1, -2, -2, -3, -3, 3, 1, -2, 1, -1, -2, -2, 0, -3, -1, 4, -3, -2, -1, -4],\n            [-2, -1, 3, 4, -3, 0, 1, -1, 0, -3, -4, 0, -3, -3, -2, 0, -1, -4, -3, -3, 4, 1, -1, -4],\n            [-1, 0, 0, 1, -3, 3, 4, -2, 0, -3, -3, 1, -1, -3, -1, 0, -1, -3, -2, -2, 1, 4, -1, -4],\n            [0, -1, -1, -1, -2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -2, 0, 0, -2, -1, -1, -1, -1, -1, -4],\n            [-4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, -4, 1]];\n        this.BLOSUM80 = [[7, -3, -3, -3, -1, -2, -2, 0, -3, -3, -3, -1, -2, -4, -1, 2, 0, -5, -4, -1, -3, -2, -1, -8],\n            [-3, 9, -1, -3, -6, 1, -1, -4, 0, -5, -4, 3, -3, -5, -3, -2, -2, -5, -4, -4, -2, 0, -2, -8],\n            [-3, -1, 9, 2, -5, 0, -1, -1, 1, -6, -6, 0, -4, -6, -4, 1, 0, -7, -4, -5, 5, -1, -2, -8],\n            [-3, -3, 2, 10, -7, -1, 2, -3, -2, -7, -7, -2, -6, -6, -3, -1, -2, -8, -6, -6, 6, 1, -3, -8],\n            [-1, -6, -5, -7, 13, -5, -7, -6, -7, -2, -3, -6, -3, -4, -6, -2, -2, -5, -5, -2, -6, -7, -4, -8],\n            [-2, 1, 0, -1, -5, 9, 3, -4, 1, -5, -4, 2, -1, -5, -3, -1, -1, -4, -3, -4, -1, 5, -2, -8],\n            [-2, -1, -1, 2, -7, 3, 8, -4, 0, -6, -6, 1, -4, -6, -2, -1, -2, -6, -5, -4, 1, 6, -2, -8],\n            [0, -4, -1, -3, -6, -4, -4, 9, -4, -7, -7, -3, -5, -6, -5, -1, -3, -6, -6, -6, -2, -4, -3, -8],\n            [-3, 0, 1, -2, -7, 1, 0, -4, 12, -6, -5, -1, -4, -2, -4, -2, -3, -4, 3, -5, -1, 0, -2, -8],\n            [-3, -5, -6, -7, -2, -5, -6, -7, -6, 7, 2, -5, 2, -1, -5, -4, -2, -5, -3, 4, -6, -6, -2, -8],\n            [-3, -4, -6, -7, -3, -4, -6, -7, -5, 2, 6, -4, 3, 0, -5, -4, -3, -4, -2, 1, -7, -5, -2, -8],\n            [-1, 3, 0, -2, -6, 2, 1, -3, -1, -5, -4, 8, -3, -5, -2, -1, -1, -6, -4, -4, -1, 1, -2, -8],\n            [-2, -3, -4, -6, -3, -1, -4, -5, -4, 2, 3, -3, 9, 0, -4, -3, -1, -3, -3, 1, -5, -3, -2, -8],\n            [-4, -5, -6, -6, -4, -5, -6, -6, -2, -1, 0, -5, 0, 10, -6, -4, -4, 0, 4, -2, -6, -6, -3, -8],\n            [-1, -3, -4, -3, -6, -3, -2, -5, -4, -5, -5, -2, -4, -6, 12, -2, -3, -7, -6, -4, -4, -2, -3, -8],\n            [2, -2, 1, -1, -2, -1, -1, -1, -2, -4, -4, -1, -3, -4, -2, 7, 2, -6, -3, -3, 0, -1, -1, -8],\n            [0, -2, 0, -2, -2, -1, -2, -3, -3, -2, -3, -1, -1, -4, -3, 2, 8, -5, -3, 0, -1, -2, -1, -8],\n            [-5, -5, -7, -8, -5, -4, -6, -6, -4, -5, -4, -6, -3, 0, -7, -6, -5, 16, 3, -5, -8, -5, -5, -8],\n            [-4, -4, -4, -6, -5, -3, -5, -6, 3, -3, -2, -4, -3, 4, -6, -3, -3, 3, 11, -3, -5, -4, -3, -8],\n            [-1, -4, -5, -6, -2, -4, -4, -6, -5, 4, 1, -4, 1, -2, -4, -3, 0, -5, -3, 7, -6, -4, -2, -8],\n            [-3, -2, 5, 6, -6, -1, 1, -2, -1, -6, -7, -1, -5, -6, -4, 0, -1, -8, -5, -6, 6, 0, -3, -8],\n            [-2, 0, -1, 1, -7, 5, 6, -4, 0, -6, -5, 1, -3, -6, -2, -1, -2, -5, -4, -4, 0, 6, -1, -8],\n            [-1, -2, -2, -3, -4, -2, -2, -3, -2, -2, -2, -2, -2, -3, -3, -1, -1, -5, -3, -2, -3, -1, -2, -8],\n            [-8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, 1]];\n        this.BLOSUM90 = [[5, -2, -2, -3, -1, -1, -1, 0, -2, -2, -2, -1, -2, -3, -1, 1, 0, -4, -3, -1, -2, -1, -1, -6],\n            [-2, 6, -1, -3, -5, 1, -1, -3, 0, -4, -3, 2, -2, -4, -3, -1, -2, -4, -3, -3, -2, 0, -2, -6],\n            [-2, -1, 7, 1, -4, 0, -1, -1, 0, -4, -4, 0, -3, -4, -3, 0, 0, -5, -3, -4, 4, -1, -2, -6],\n            [-3, -3, 1, 7, -5, -1, 1, -2, -2, -5, -5, -1, -4, -5, -3, -1, -2, -6, -4, -5, 4, 0, -2, -6],\n            [-1, -5, -4, -5, 9, -4, -6, -4, -5, -2, -2, -4, -2, -3, -4, -2, -2, -4, -4, -2, -4, -5, -3, -6],\n            [-1, 1, 0, -1, -4, 7, 2, -3, 1, -4, -3, 1, 0, -4, -2, -1, -1, -3, -3, -3, -1, 4, -1, -6],\n            [-1, -1, -1, 1, -6, 2, 6, -3, -1, -4, -4, 0, -3, -5, -2, -1, -1, -5, -4, -3, 0, 4, -2, -6],\n            [0, -3, -1, -2, -4, -3, -3, 6, -3, -5, -5, -2, -4, -5, -3, -1, -3, -4, -5, -5, -2, -3, -2, -6],\n            [-2, 0, 0, -2, -5, 1, -1, -3, 8, -4, -4, -1, -3, -2, -3, -2, -2, -3, 1, -4, -1, 0, -2, -6],\n            [-2, -4, -4, -5, -2, -4, -4, -5, -4, 5, 1, -4, 1, -1, -4, -3, -1, -4, -2, 3, -5, -4, -2, -6],\n            [-2, -3, -4, -5, -2, -3, -4, -5, -4, 1, 5, -3, 2, 0, -4, -3, -2, -3, -2, 0, -5, -4, -2, -6],\n            [-1, 2, 0, -1, -4, 1, 0, -2, -1, -4, -3, 6, -2, -4, -2, -1, -1, -5, -3, -3, -1, 1, -1, -6],\n            [-2, -2, -3, -4, -2, 0, -3, -4, -3, 1, 2, -2, 7, -1, -3, -2, -1, -2, -2, 0, -4, -2, -1, -6],\n            [-3, -4, -4, -5, -3, -4, -5, -5, -2, -1, 0, -4, -1, 7, -4, -3, -3, 0, 3, -2, -4, -4, -2, -6],\n            [-1, -3, -3, -3, -4, -2, -2, -3, -3, -4, -4, -2, -3, -4, 8, -2, -2, -5, -4, -3, -3, -2, -2, -6],\n            [1, -1, 0, -1, -2, -1, -1, -1, -2, -3, -3, -1, -2, -3, -2, 5, 1, -4, -3, -2, 0, -1, -1, -6],\n            [0, -2, 0, -2, -2, -1, -1, -3, -2, -1, -2, -1, -1, -3, -2, 1, 6, -4, -2, -1, -1, -1, -1, -6],\n            [-4, -4, -5, -6, -4, -3, -5, -4, -3, -4, -3, -5, -2, 0, -5, -4, -4, 11, 2, -3, -6, -4, -3, -6],\n            [-3, -3, -3, -4, -4, -3, -4, -5, 1, -2, -2, -3, -2, 3, -4, -3, -2, 2, 8, -3, -4, -3, -2, -6],\n            [-1, -3, -4, -5, -2, -3, -3, -5, -4, 3, 0, -3, 0, -2, -3, -2, -1, -3, -3, 5, -4, -3, -2, -6],\n            [-2, -2, 4, 4, -4, -1, 0, -2, -1, -5, -5, -1, -4, -4, -3, 0, -1, -6, -4, -4, 4, 0, -2, -6],\n            [-1, 0, -1, 0, -5, 4, 4, -3, 0, -4, -4, 1, -2, -4, -2, -1, -1, -4, -3, -3, 0, 4, -1, -6],\n            [-1, -2, -2, -2, -3, -1, -2, -2, -2, -2, -2, -1, -1, -2, -2, -1, -1, -3, -2, -2, -2, -1, -2, -6],\n            [-6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, -6, 1]];\n        this.NUCLEOTIDES = [[1, -1, -1, -1],\n            [-1, 1, -1, -1],\n            [-1, -1, 1, -1],\n            [-1, -1, -1, 1]];\n        this.PAM30 = [[6, -7, -4, -3, -6, -4, -2, -2, -7, -5, -6, -7, -5, -8, -2, 0, -1, -13, -8, -2, -3, -3, -3, -17],\n            [-7, 8, -6, -10, -8, -2, -9, -9, -2, -5, -8, 0, -4, -9, -4, -3, -6, -2, -10, -8, -7, -4, -6, -17],\n            [-4, -6, 8, 2, -11, -3, -2, -3, 0, -5, -7, -1, -9, -9, -6, 0, -2, -8, -4, -8, 6, -3, -3, -17],\n            [-3, -10, 2, 8, -14, -2, 2, -3, -4, -7, -12, -4, -11, -15, -8, -4, -5, -15, -11, -8, 6, 1, -5, -17],\n            [-6, -8, -11, -14, 10, -14, -14, -9, -7, -6, -15, -14, -13, -13, -8, -3, -8, -15, -4, -6, -12, -14, -9, -17],\n            [-4, -2, -3, -2, -14, 8, 1, -7, 1, -8, -5, -3, -4, -13, -3, -5, -5, -13, -12, -7, -3, 6, -5, -17],\n            [-2, -9, -2, 2, -14, 1, 8, -4, -5, -5, -9, -4, -7, -14, -5, -4, -6, -17, -8, -6, 1, 6, -5, -17],\n            [-2, -9, -3, -3, -9, -7, -4, 6, -9, -11, -10, -7, -8, -9, -6, -2, -6, -15, -14, -5, -3, -5, -5, -17],\n            [-7, -2, 0, -4, -7, 1, -5, -9, 9, -9, -6, -6, -10, -6, -4, -6, -7, -7, -3, -6, -1, -1, -5, -17],\n            [-5, -5, -5, -7, -6, -8, -5, -11, -9, 8, -1, -6, -1, -2, -8, -7, -2, -14, -6, 2, -6, -6, -5, -17],\n            [-6, -8, -7, -12, -15, -5, -9, -10, -6, -1, 7, -8, 1, -3, -7, -8, -7, -6, -7, -2, -9, -7, -6, -17],\n            [-7, 0, -1, -4, -14, -3, -4, -7, -6, -6, -8, 7, -2, -14, -6, -4, -3, -12, -9, -9, -2, -4, -5, -17],\n            [-5, -4, -9, -11, -13, -4, -7, -8, -10, -1, 1, -2, 11, -4, -8, -5, -4, -13, -11, -1, -10, -5, -5, -17],\n            [-8, -9, -9, -15, -13, -13, -14, -9, -6, -2, -3, -14, -4, 9, -10, -6, -9, -4, 2, -8, -10, -13, -8, -17],\n            [-2, -4, -6, -8, -8, -3, -5, -6, -4, -8, -7, -6, -8, -10, 8, -2, -4, -14, -13, -6, -7, -4, -5, -17],\n            [0, -3, 0, -4, -3, -5, -4, -2, -6, -7, -8, -4, -5, -6, -2, 6, 0, -5, -7, -6, -1, -5, -3, -17],\n            [-1, -6, -2, -5, -8, -5, -6, -6, -7, -2, -7, -3, -4, -9, -4, 0, 7, -13, -6, -3, -3, -6, -4, -17],\n            [-13, -2, -8, -15, -15, -13, -17, -15, -7, -14, -6, -12, -13, -4, -14, -5, -13, 13, -5, -15, -10, -14, -11, -17],\n            [-8, -10, -4, -11, -4, -12, -8, -14, -3, -6, -7, -9, -11, 2, -13, -7, -6, -5, 10, -7, -6, -9, -7, -17],\n            [-2, -8, -8, -8, -6, -7, -6, -5, -6, 2, -2, -9, -1, -8, -6, -6, -3, -15, -7, 7, -8, -6, -5, -17],\n            [-3, -7, 6, 6, -12, -3, 1, -3, -1, -6, -9, -2, -10, -10, -7, -1, -3, -10, -6, -8, 6, 0, -5, -17],\n            [-3, -4, -3, 1, -14, 6, 6, -5, -1, -6, -7, -4, -5, -13, -4, -5, -6, -14, -9, -6, 0, 6, -5, -17],\n            [-3, -6, -3, -5, -9, -5, -5, -5, -5, -5, -6, -5, -5, -8, -5, -3, -4, -11, -7, -5, -5, -5, -5, -17],\n            [-17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, -17, 1]];\n        this.PAM70 = [[5, -4, -2, -1, -4, -2, -1, 0, -4, -2, -4, -4, -3, -6, 0, 1, 1, -9, -5, -1, -1, -1, -2, -11],\n            [-4, 8, -3, -6, -5, 0, -5, -6, 0, -3, -6, 2, -2, -7, -2, -1, -4, 0, -7, -5, -4, -2, -3, -11],\n            [-2, -3, 6, 3, -7, -1, 0, -1, 1, -3, -5, 0, -5, -6, -3, 1, 0, -6, -3, -5, 5, -1, -2, -11],\n            [-1, -6, 3, 6, -9, 0, 3, -1, -1, -5, -8, -2, -7, -10, -4, -1, -2, -10, -7, -5, 5, 2, -3, -11],\n            [-4, -5, -7, -9, 9, -9, -9, -6, -5, -4, -10, -9, -9, -8, -5, -1, -5, -11, -2, -4, -8, -9, -6, -11],\n            [-2, 0, -1, 0, -9, 7, 2, -4, 2, -5, -3, -1, -2, -9, -1, -3, -3, -8, -8, -4, -1, 5, -2, -11],\n            [-1, -5, 0, 3, -9, 2, 6, -2, -2, -4, -6, -2, -4, -9, -3, -2, -3, -11, -6, -4, 2, 5, -3, -11],\n            [0, -6, -1, -1, -6, -4, -2, 6, -6, -6, -7, -5, -6, -7, -3, 0, -3, -10, -9, -3, -1, -3, -3, -11],\n            [-4, 0, 1, -1, -5, 2, -2, -6, 8, -6, -4, -3, -6, -4, -2, -3, -4, -5, -1, -4, 0, 1, -3, -11],\n            [-2, -3, -3, -5, -4, -5, -4, -6, -6, 7, 1, -4, 1, 0, -5, -4, -1, -9, -4, 3, -4, -4, -3, -11],\n            [-4, -6, -5, -8, -10, -3, -6, -7, -4, 1, 6, -5, 2, -1, -5, -6, -4, -4, -4, 0, -6, -4, -4, -11],\n            [-4, 2, 0, -2, -9, -1, -2, -5, -3, -4, -5, 6, 0, -9, -4, -2, -1, -7, -7, -6, -1, -2, -3, -11],\n            [-3, -2, -5, -7, -9, -2, -4, -6, -6, 1, 2, 0, 10, -2, -5, -3, -2, -8, -7, 0, -6, -3, -3, -11],\n            [-6, -7, -6, -10, -8, -9, -9, -7, -4, 0, -1, -9, -2, 8, -7, -4, -6, -2, 4, -5, -7, -9, -5, -11],\n            [0, -2, -3, -4, -5, -1, -3, -3, -2, -5, -5, -4, -5, -7, 7, 0, -2, -9, -9, -3, -4, -2, -3, -11],\n            [1, -1, 1, -1, -1, -3, -2, 0, -3, -4, -6, -2, -3, -4, 0, 5, 2, -3, -5, -3, 0, -2, -1, -11],\n            [1, -4, 0, -2, -5, -3, -3, -3, -4, -1, -4, -1, -2, -6, -2, 2, 6, -8, -4, -1, -1, -3, -2, -11],\n            [-9, 0, -6, -10, -11, -8, -11, -10, -5, -9, -4, -7, -8, -2, -9, -3, -8, 13, -3, -10, -7, -10, -7, -11],\n            [-5, -7, -3, -7, -2, -8, -6, -9, -1, -4, -4, -7, -7, 4, -9, -5, -4, -3, 9, -5, -4, -7, -5, -11],\n            [-1, -5, -5, -5, -4, -4, -4, -3, -4, 3, 0, -6, 0, -5, -3, -3, -1, -10, -5, 6, -5, -4, -2, -11],\n            [-1, -4, 5, 5, -8, -1, 2, -1, 0, -4, -6, -1, -6, -7, -4, 0, -1, -7, -4, -5, 5, 1, -2, -11],\n            [-1, -2, -1, 2, -9, 5, 5, -3, 1, -4, -4, -2, -3, -9, -2, -2, -3, -10, -7, -4, 1, 5, -3, -11],\n            [-2, -3, -2, -3, -6, -2, -3, -3, -3, -3, -4, -3, -3, -5, -3, -1, -2, -7, -5, -2, -2, -3, -3, -11],\n            [-11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, -11, 1]];\n        this.PAM250 = [[2, -2, 0, 0, -2, 0, 0, 1, -1, -1, -2, -1, -1, -3, 1, 1, 1, -6, -3, 0, 0, 0, 0, -8],\n            [-2, 6, 0, -1, -4, 1, -1, -3, 2, -2, -3, 3, 0, -4, 0, 0, -1, 2, -4, -2, -1, 0, -1, -8],\n            [0, 0, 2, 2, -4, 1, 1, 0, 2, -2, -3, 1, -2, -3, 0, 1, 0, -4, -2, -2, 2, 1, 0, -8],\n            [0, -1, 2, 4, -5, 2, 3, 1, 1, -2, -4, 0, -3, -6, -1, 0, 0, -7, -4, -2, 3, 3, -1, -8],\n            [-2, -4, -4, -5, 12, -5, -5, -3, -3, -2, -6, -5, -5, -4, -3, 0, -2, -8, 0, -2, -4, -5, -3, -8],\n            [0, 1, 1, 2, -5, 4, 2, -1, 3, -2, -2, 1, -1, -5, 0, -1, -1, -5, -4, -2, 1, 3, -1, -8],\n            [0, -1, 1, 3, -5, 2, 4, 0, 1, -2, -3, 0, -2, -5, -1, 0, 0, -7, -4, -2, 3, 3, -1, -8],\n            [1, -3, 0, 1, -3, -1, 0, 5, -2, -3, -4, -2, -3, -5, 0, 1, 0, -7, -5, -1, 0, 0, -1, -8],\n            [-1, 2, 2, 1, -3, 3, 1, -2, 6, -2, -2, 0, -2, -2, 0, -1, -1, -3, 0, -2, 1, 2, -1, -8],\n            [-1, -2, -2, -2, -2, -2, -2, -3, -2, 5, 2, -2, 2, 1, -2, -1, 0, -5, -1, 4, -2, -2, -1, -8],\n            [-2, -3, -3, -4, -6, -2, -3, -4, -2, 2, 6, -3, 4, 2, -3, -3, -2, -2, -1, 2, -3, -3, -1, -8],\n            [-1, 3, 1, 0, -5, 1, 0, -2, 0, -2, -3, 5, 0, -5, -1, 0, 0, -3, -4, -2, 1, 0, -1, -8],\n            [-1, 0, -2, -3, -5, -1, -2, -3, -2, 2, 4, 0, 6, 0, -2, -2, -1, -4, -2, 2, -2, -2, -1, -8],\n            [-3, -4, -3, -6, -4, -5, -5, -5, -2, 1, 2, -5, 0, 9, -5, -3, -3, 0, 7, -1, -4, -5, -2, -8],\n            [1, 0, 0, -1, -3, 0, -1, 0, 0, -2, -3, -1, -2, -5, 6, 1, 0, -6, -5, -1, -1, 0, -1, -8],\n            [1, 0, 1, 0, 0, -1, 0, 1, -1, -1, -3, 0, -2, -3, 1, 2, 1, -2, -3, -1, 0, 0, 0, -8],\n            [1, -1, 0, 0, -2, -1, 0, 0, -1, 0, -2, 0, -1, -3, 0, 1, 3, -5, -3, 0, 0, -1, 0, -8],\n            [-6, 2, -4, -7, -8, -5, -7, -7, -3, -5, -2, -3, -4, 0, -6, -2, -5, 17, 0, -6, -5, -6, -4, -8],\n            [-3, -4, -2, -4, 0, -4, -4, -5, 0, -1, -1, -4, -2, 7, -5, -3, -3, 0, 10, -2, -3, -4, -2, -8],\n            [0, -2, -2, -2, -2, -2, -2, -1, -2, 4, 2, -2, 2, -1, -1, -1, 0, -6, -2, 4, -2, -2, -1, -8],\n            [0, -1, 2, 3, -4, 1, 3, 0, 1, -2, -3, 1, -2, -4, -1, 0, 0, -5, -3, -2, 3, 2, -1, -8],\n            [0, 0, 1, 3, -5, 3, 3, 0, 2, -2, -3, 0, -2, -5, 0, 0, -1, -6, -4, -2, 2, 3, -1, -8],\n            [0, -1, 0, -1, -3, -1, -1, -1, -1, -1, -1, -1, -1, -2, -1, 0, 0, -4, -2, -1, -1, -1, -1, -8],\n            [-8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, -8, 1]];\n        this.SCHNEIDER = [[11.6, -2.7, 9.7, -1.7, -2.7, -6.4, -3.9, -5.6, 5.1, -5.0, 3.6, -4.2, -6.3, -13.0, -7.1, -11.5, 0.4, -6.0, -1.9, -5.3, -8.5, -11.2, -8.9, -10.8, 2.1, 0.0, 1.4, 0.2, -10.2, -13.5, -13.0, -12.5, -2.6, -8.5, -5.0, -8.1, -6.3, -9.9, -7.5, -9.0, -7.1, -10.2, -8.2, -9.2, -8.2, -12.5, -11.1, -11.4, -50.0, -14.8, -50.0, -13.8, -7.3, -10.1, -8.4, -9.1, -50.0, -13.0, -13.5, -12.4, -10.7, -18.1, -11.8, -17.2,],\n            [-2.7, 13.0, -3.3, 10.9, -3.5, -0.4, -3.3, -1.8, -5.4, 4.6, -5.5, 3.0, -10.2, -7.9, -9.9, -9.6, -5.0, 0.5, -5.5, -1.0, -10.3, -8.1, -9.4, -9.6, -8.1, -5.0, -7.3, -6.3, -13.4, -11.3, -14.4, -12.9, -6.3, 0.8, -6.4, -1.1, -7.4, -5.0, -6.2, -6.5, -5.6, -1.6, -4.7, -3.0, -10.8, -8.7, -11.9, -10.0, -50.0, -6.2, -50.0, -7.5, -6.3, -4.3, -6.2, -5.4, -50.0, -7.0, -16.3, -8.2, -13.2, -12.3, -13.1, -13.3,],\n            [9.7, -3.3, 11.6, -2.8, -4.5, -6.7, -3.1, -6.9, 3.3, -5.5, 4.8, -5.1, -8.9, -13.2, -5.7, -12.6, -1.5, -6.1, -0.6, -6.1, -10.0, -11.8, -8.6, -11.9, 1.2, 0.5, 2.2, 0.1, -11.8, -14.0, -11.9, -13.4, -4.9, -9.1, -3.4, -8.9, -8.0, -10.0, -7.2, -10.0, -9.1, -10.2, -7.1, -9.9, -10.1, -13.0, -10.6, -12.7, -50.0, -14.9, -50.0, -14.4, -8.9, -10.8, -8.9, -10.4, -50.0, -13.1, -11.8, -13.0, -12.4, -19.4, -11.5, -17.8,],\n            [-1.7, 10.9, -2.8, 12.9, -2.7, -2.2, -2.8, 0.2, -4.8, 2.9, -5.2, 5.2, -9.1, -9.5, -9.0, -7.0, -4.0, -1.0, -5.0, 1.0, -9.0, -8.8, -8.9, -7.7, -7.0, -6.8, -7.1, -4.3, -12.4, -12.9, -13.9, -10.0, -4.9, -0.7, -5.6, 1.4, -6.0, -6.5, -6.2, -4.8, -5.1, -3.2, -5.0, -1.2, -9.8, -9.9, -11.3, -8.2, -50.0, -7.6, -50.0, -5.1, -5.5, -5.8, -5.9, -4.2, -50.0, -8.3, -15.5, -5.9, -11.6, -14.2, -12.0, -11.6,],\n            [-2.7, -3.5, -4.5, -2.7, 11.7, 9.0, 10.6, 9.6, -3.2, -0.7, -5.1, -0.4, 0.7, -4.6, -0.4, -3.6, -4.8, -8.9, -6.5, -7.8, -1.4, -4.1, -2.9, -3.3, -8.0, -9.1, -8.0, -8.2, -6.0, -9.3, -7.7, -8.5, -6.4, -10.1, -7.5, -8.6, 2.8, -0.7, 0.9, 0.1, -5.9, -7.3, -6.5, -6.5, -0.4, -3.9, -2.6, -3.1, -50.0, -14.6, -50.0, -12.6, 2.9, 0.1, 1.8, 1.0, -50.0, -9.2, -13.6, -7.7, -4.8, -12.2, -6.2, -11.0,],\n            [-6.4, -0.4, -6.7, -2.2, 9.0, 12.3, 9.8, 9.6, -7.1, 2.4, -7.2, 0.3, -2.9, -1.2, -3.0, -3.3, -7.6, -6.7, -8.1, -7.9, -4.9, -1.9, -4.5, -3.8, -9.5, -7.8, -9.3, -8.9, -9.4, -7.1, -9.0, -9.0, -9.4, -7.5, -9.3, -9.2, -0.7, 2.6, 0.2, -0.1, -8.0, -4.6, -7.7, -6.1, -3.6, -1.0, -4.1, -3.1, -50.0, -11.6, -50.0, -11.6, 0.2, 2.3, 0.4, 0.5, -50.0, -6.5, -15.3, -7.8, -8.4, -9.5, -8.3, -11.1,],\n            [-3.9, -3.3, -3.1, -2.8, 10.6, 9.8, 12.2, 9.8, -4.9, 0.1, -3.6, -0.2, -1.7, -4.1, 0.9, -3.5, -5.9, -8.9, -5.6, -8.5, -3.9, -3.9, -2.2, -4.5, -7.3, -7.5, -6.4, -8.5, -7.1, -8.2, -6.7, -7.8, -7.9, -9.1, -7.1, -9.4, 0.9, -0.2, 2.5, -0.3, -6.9, -6.5, -5.5, -6.8, -1.7, -3.4, -1.7, -2.8, -50.0, -13.0, -50.0, -11.5, 1.5, 0.7, 2.7, 0.6, -50.0, -8.9, -12.1, -7.4, -6.1, -11.5, -5.4, -10.4,],\n            [-5.6, -1.8, -6.9, 0.2, 9.6, 9.6, 9.8, 11.6, -6.6, 0.9, -7.2, 2.5, -2.3, -3.4, -2.3, -0.6, -6.5, -8.2, -8.0, -6.3, -3.7, -3.5, -3.9, -1.6, -9.9, -8.9, -9.6, -7.0, -8.6, -8.9, -8.8, -6.8, -8.6, -8.2, -8.8, -6.8, 0.3, 0.1, 0.2, 2.4, -7.4, -6.2, -6.9, -4.4, -2.9, -2.9, -3.5, -0.9, -50.0, -12.2, -50.0, -10.1, 1.2, 0.6, 1.2, 2.4, -50.0, -7.6, -16.1, -5.8, -7.2, -10.8, -7.4, -9.0,],\n            [5.1, -5.4, 3.3, -4.8, -3.2, -7.1, -4.9, -6.6, 13.3, -2.8, 11.2, -1.9, -5.7, -12.5, -7.2, -11.6, -0.7, -4.8, -3.1, -4.3, -10.0, -11.6, -8.8, -11.7, 10.5, 7.7, 9.1, 8.5, -9.8, -12.7, -11.7, -11.8, -6.3, -11.4, -8.8, -11.0, -7.5, -10.3, -8.3, -9.9, -1.9, -6.7, -4.1, -6.4, -7.9, -12.5, -11.4, -11.7, -50.0, -14.0, -50.0, -13.2, -8.5, -10.4, -9.1, -9.6, -50.0, -9.8, -7.7, -8.8, -10.3, -17.9, -11.8, -16.0,],\n            [-5.0, 4.6, -5.5, 2.9, -0.7, 2.4, 0.1, 0.9, -2.8, 12.8, -2.6, 11.0, -8.4, -6.3, -8.3, -7.7, -6.1, -2.9, -6.5, -4.2, -8.5, -5.9, -7.5, -7.7, -5.5, -2.4, -5.5, -4.1, -12.9, -10.3, -12.7, -11.3, -7.4, -3.1, -7.4, -4.9, -4.4, -1.8, -3.5, -3.5, -1.6, 3.0, -0.8, 0.9, -8.4, -6.3, -9.6, -7.3, -50.0, -9.0, -50.0, -9.7, -2.3, -0.2, -1.2, -1.5, -50.0, -0.7, -13.3, -2.0, -11.8, -11.9, -12.3, -12.7,],\n            [3.6, -5.5, 4.8, -5.2, -5.1, -7.2, -3.6, -7.2, 11.2, -2.6, 13.4, -2.1, -7.7, -12.5, -5.3, -11.9, -2.2, -4.4, -1.9, -4.8, -10.2, -11.5, -8.1, -12.0, 9.3, 8.2, 10.0, 8.0, -10.8, -11.9, -10.7, -12.4, -8.2, -11.8, -6.9, -11.5, -8.4, -9.9, -7.1, -10.6, -4.7, -6.8, -1.2, -7.0, -9.8, -11.6, -9.6, -12.1, -50.0, -14.1, -50.0, -12.7, -10.0, -10.8, -8.8, -10.5, -50.0, -9.2, -4.2, -9.3, -11.0, -18.2, -11.1, -16.1,],\n            [-4.2, 3.0, -5.1, 5.2, -0.4, 0.3, -0.2, 2.5, -1.9, 11.0, -2.1, 13.2, -7.6, -8.5, -8.0, -5.3, -5.6, -4.5, -6.8, -2.4, -7.8, -7.9, -7.9, -6.2, -5.7, -5.1, -5.4, -2.1, -12.4, -12.3, -13.1, -10.1, -6.6, -4.4, -6.9, -2.6, -3.6, -3.8, -3.7, -2.0, -1.4, 0.8, -0.9, 3.0, -8.2, -8.1, -9.2, -5.8, -50.0, -10.1, -50.0, -7.4, -1.7, -2.0, -1.5, -0.5, -50.0, -2.1, -12.6, -0.4, -11.6, -13.9, -11.4, -11.1,],\n            [-6.3, -10.2, -8.9, -9.1, 0.7, -2.9, -1.7, -2.3, -5.7, -8.4, -7.7, -7.6, 13.2, 9.6, 3.5, 9.7, -8.7, -12.4, -10.7, -10.7, -7.8, -10.2, -8.8, -9.6, -9.5, -11.9, -10.9, -10.5, 2.3, -0.4, -0.2, -0.3, -9.7, -15.5, -11.4, -13.7, -3.0, -6.4, -4.1, -5.4, -9.6, -11.9, -10.5, -11.8, 6.2, 3.3, 3.7, 3.6, -50.0, -13.6, -50.0, -11.9, -5.6, -8.8, -7.2, -8.9, -50.0, -12.4, -14.1, -11.6, 2.8, -6.4, 0.5, -5.2,],\n            [-13.0, -7.9, -13.2, -9.5, -4.6, -1.2, -4.1, -3.4, -12.5, -6.3, -12.5, -8.5, 9.6, 12.7, 0.2, 10.5, -12.4, -11.6, -13.2, -12.0, -12.4, -10.1, -11.6, -12.0, -15.1, -13.1, -14.3, -13.4, -1.4, 1.4, -1.6, -0.5, -14.9, -14.3, -15.2, -16.8, -7.7, -4.7, -6.3, -6.6, -14.6, -10.8, -12.8, -13.0, 2.3, 6.0, 2.5, 3.6, -50.0, -11.2, -50.0, -11.9, -10.5, -9.2, -10.5, -10.5, -50.0, -10.3, -16.2, -11.3, -1.7, -3.7, -2.5, -5.4,],\n            [-7.1, -9.9, -5.7, -9.0, -0.4, -3.0, 0.9, -2.3, -7.2, -8.3, -5.3, -8.0, 3.5, 0.2, 14.3, 1.0, -7.3, -11.2, -6.9, -9.7, -8.5, -10.2, -7.7, -9.8, -10.0, -10.4, -8.4, -10.2, 1.0, -0.6, 1.7, -0.5, -11.1, -14.8, -9.7, -14.5, -3.7, -5.8, -3.1, -5.6, -10.4, -11.5, -8.2, -11.1, 0.6, -1.6, 1.6, -1.1, -50.0, -12.4, -50.0, -11.5, -5.5, -8.0, -4.9, -7.2, -50.0, -12.4, -10.3, -11.6, 0.7, -6.8, 2.4, -6.0,],\n            [-11.5, -9.6, -12.6, -7.0, -3.6, -3.3, -3.5, -0.6, -11.6, -7.7, -11.9, -5.3, 9.7, 10.5, 1.0, 12.6, -12.0, -12.2, -12.5, -9.7, -11.1, -11.2, -11.9, -9.4, -12.9, -14.3, -14.0, -11.9, -1.0, -0.4, -1.6, 1.2, -13.9, -15.4, -13.7, -12.6, -6.2, -6.7, -6.3, -4.4, -13.0, -12.2, -13.0, -10.4, 2.9, 3.8, 2.5, 5.8, -50.0, -12.5, -50.0, -10.2, -9.6, -9.7, -9.6, -8.3, -50.0, -11.6, -15.4, -9.4, -1.1, -5.4, -1.6, -3.3,],\n            [0.4, -5.0, -1.5, -4.0, -4.8, -7.6, -5.9, -6.5, -0.7, -6.1, -2.2, -5.6, -8.7, -12.4, -7.3, -12.0, 12.8, 2.3, 10.2, 3.0, 0.0, -3.4, -0.7, -3.2, 2.5, -0.8, 0.9, 0.2, -3.0, -7.2, -5.7, -6.1, -0.2, -6.6, -1.8, -6.0, -5.3, -8.2, -5.7, -7.6, -6.7, -9.6, -7.5, -9.3, -7.1, -10.5, -9.6, -9.4, -50.0, -8.1, -50.0, -7.2, -4.3, -6.8, -5.7, -6.3, -50.0, -9.5, -9.6, -8.9, -6.2, -12.8, -6.8, -11.8,],\n            [-6.0, 0.5, -6.1, -1.0, -8.9, -6.7, -8.9, -8.2, -4.8, -2.9, -4.4, -4.5, -12.4, -11.6, -11.2, -12.2, 2.3, 14.6, 1.9, 12.9, -5.5, -2.6, -4.1, -4.7, -1.2, 3.0, -1.1, 1.6, -8.0, -4.7, -8.7, -6.0, -8.2, -4.7, -7.8, -6.7, -10.0, -8.3, -8.9, -9.9, -10.8, -7.1, -10.0, -9.2, -11.6, -10.1, -12.3, -11.9, -50.0, 2.2, -50.0, 0.9, -7.9, -5.9, -7.4, -6.9, -50.0, -4.6, -11.6, -5.6, -9.7, -5.9, -9.9, -7.3,],\n            [-1.9, -5.5, -0.6, -5.0, -6.5, -8.1, -5.6, -8.0, -3.1, -6.5, -1.9, -6.8, -10.7, -13.2, -6.9, -12.5, 10.2, 1.9, 11.9, 2.1, -2.3, -4.1, -0.5, -4.6, -0.1, -0.9, 2.4, -0.8, -5.1, -6.9, -4.8, -6.7, -2.3, -7.1, -0.8, -7.2, -7.2, -8.0, -5.6, -8.5, -9.4, -9.7, -7.2, -9.8, -9.2, -11.1, -9.5, -10.7, -50.0, -8.7, -50.0, -8.6, -6.1, -7.5, -5.5, -7.7, -50.0, -10.4, -7.7, -10.0, -7.3, -13.6, -6.4, -13.3,],\n            [-5.3, -1.0, -6.1, 1.0, -7.8, -7.9, -8.5, -6.3, -4.3, -4.2, -4.8, -2.4, -10.7, -12.0, -9.7, -9.7, 3.0, 12.9, 2.1, 14.7, -4.9, -3.8, -4.0, -2.4, -1.0, 0.8, -1.3, 3.4, -7.6, -5.7, -8.1, -3.5, -6.8, -5.8, -7.2, -4.2, -8.9, -10.2, -8.4, -8.2, -9.9, -8.4, -9.7, -6.4, -11.1, -11.1, -11.8, -10.0, -50.0, 0.4, -50.0, 2.6, -7.2, -7.0, -7.4, -5.1, -50.0, -5.7, -9.8, -3.2, -8.4, -7.4, -8.9, -5.9,],\n            [-8.5, -10.3, -10.0, -9.0, -1.4, -4.9, -3.9, -3.7, -10.0, -8.5, -10.2, -7.8, -7.8, -12.4, -8.5, -11.1, 0.0, -5.5, -2.3, -4.9, 12.6, 10.0, 11.1, 10.5, -6.0, -8.3, -6.9, -7.5, -2.2, -7.4, -5.3, -5.9, -8.9, -12.5, -9.4, -11.6, -0.7, -3.9, -2.3, -3.0, -9.1, -9.8, -9.2, -9.5, -5.6, -9.0, -8.0, -8.5, -50.0, -15.3, -50.0, -14.1, 2.3, -1.4, 0.3, -0.5, -50.0, -13.6, -14.3, -11.5, -5.3, -13.7, -6.1, -12.7,],\n            [-11.2, -8.1, -11.8, -8.8, -4.1, -1.9, -3.9, -3.5, -11.6, -5.9, -11.5, -7.9, -10.2, -10.1, -10.2, -11.2, -3.4, -2.6, -4.1, -3.8, 10.0, 13.1, 10.7, 10.6, -8.1, -5.2, -8.0, -6.7, -6.1, -3.7, -7.5, -5.6, -11.1, -10.3, -10.6, -11.9, -3.2, -1.0, -2.2, -2.8, -10.5, -8.1, -9.4, -9.5, -8.3, -6.9, -9.4, -8.8, -50.0, -11.3, -50.0, -12.8, -0.6, 2.2, -0.3, 0.1, -50.0, -10.1, -17.5, -11.1, -8.6, -9.9, -8.6, -11.9,],\n            [-8.9, -9.4, -8.6, -8.9, -2.9, -4.5, -2.2, -3.9, -8.8, -7.5, -8.1, -7.9, -8.8, -11.6, -7.7, -11.9, -0.7, -4.1, -0.5, -4.0, 11.1, 10.7, 13.2, 10.4, -5.7, -5.8, -3.5, -6.1, -3.6, -6.5, -3.3, -5.5, -9.4, -11.0, -8.7, -11.2, -1.8, -2.7, 0.7, -3.0, -9.1, -8.2, -7.3, -9.4, -7.2, -8.2, -7.2, -8.2, -50.0, -13.2, -50.0, -13.3, 0.5, -0.9, 1.8, -0.6, -50.0, -11.4, -11.1, -10.5, -6.1, -12.9, -5.0, -11.9,],\n            [-10.8, -9.6, -11.9, -7.7, -3.3, -3.8, -4.5, -1.6, -11.7, -7.7, -12.0, -6.2, -9.6, -12.0, -9.8, -9.4, -3.2, -4.7, -4.6, -2.4, 10.5, 10.6, 10.4, 12.6, -8.2, -7.9, -9.0, -4.7, -6.1, -6.0, -7.5, -2.9, -10.9, -12.2, -11.2, -10.9, -2.4, -2.9, -2.7, -0.9, -10.4, -9.7, -10.2, -8.0, -8.4, -9.1, -9.3, -6.5, -50.0, -13.3, -50.0, -11.0, -0.0, -0.4, -0.4, 2.3, -50.0, -10.9, -17.2, -8.3, -8.3, -12.7, -7.8, -9.3,],\n            [2.1, -8.1, 1.2, -7.0, -8.0, -9.5, -7.3, -9.9, 10.5, -5.5, 9.3, -5.7, -9.5, -15.1, -10.0, -12.9, 2.5, -1.2, -0.1, -1.0, -6.0, -8.1, -5.7, -8.2, 13.8, 11.3, 11.8, 12.1, -6.0, -9.1, -9.0, -9.5, -8.9, -13.4, -10.0, -13.1, -9.9, -11.7, -9.8, -12.8, -5.9, -9.4, -6.5, -8.1, -10.7, -14.0, -13.3, -13.9, -50.0, -11.9, -50.0, -9.7, -8.9, -10.8, -9.0, -11.1, -50.0, -6.5, -5.3, -6.3, -10.5, -16.2, -10.4, -15.2,],\n            [0.0, -5.0, 0.5, -6.8, -9.1, -7.8, -7.5, -8.9, 7.7, -2.4, 8.2, -5.1, -11.9, -13.1, -10.4, -14.3, -0.8, 3.0, -0.9, 0.8, -8.3, -5.2, -5.8, -7.9, 11.3, 15.0, 11.2, 12.8, -9.7, -5.3, -9.2, -7.8, -11.7, -10.6, -10.1, -12.2, -11.7, -8.8, -8.9, -11.8, -9.3, -5.0, -8.0, -8.2, -13.6, -11.4, -12.8, -13.0, -50.0, -7.2, -50.0, -8.4, -11.0, -7.7, -8.7, -10.2, -50.0, -1.5, -7.1, -4.9, -11.2, -11.6, -11.6, -15.0,],\n            [1.4, -7.3, 2.2, -7.1, -8.0, -9.3, -6.4, -9.6, 9.1, -5.5, 10.0, -5.4, -10.9, -14.3, -8.4, -14.0, 0.9, -1.1, 2.4, -1.3, -6.9, -8.0, -3.5, -9.0, 11.8, 11.2, 13.4, 11.4, -7.3, -8.8, -6.5, -8.5, -10.0, -12.5, -7.8, -12.5, -9.6, -11.0, -7.1, -10.7, -7.9, -8.0, -4.3, -9.4, -11.4, -12.3, -10.9, -12.9, -50.0, -11.8, -50.0, -11.2, -9.4, -9.8, -7.3, -10.5, -50.0, -6.2, -2.2, -6.7, -9.8, -15.2, -8.5, -14.9,],\n            [0.2, -6.3, 0.1, -4.3, -8.2, -8.9, -8.5, -7.0, 8.5, -4.1, 8.0, -2.1, -10.5, -13.4, -10.2, -11.9, 0.2, 1.6, -0.8, 3.4, -7.5, -6.7, -6.1, -4.7, 12.1, 12.8, 11.4, 14.7, -8.7, -7.4, -9.4, -5.3, -9.9, -11.8, -10.5, -10.3, -9.9, -10.1, -9.3, -9.3, -8.1, -7.2, -8.0, -5.2, -12.0, -12.1, -12.6, -10.9, -50.0, -7.9, -50.0, -5.3, -9.3, -8.8, -8.9, -7.0, -50.0, -3.6, -7.1, -1.2, -10.2, -12.7, -9.9, -11.0,],\n            [-10.2, -13.4, -11.8, -12.4, -6.0, -9.4, -7.1, -8.6, -9.8, -12.9, -10.8, -12.4, 2.3, -1.4, 1.0, -1.0, -3.0, -8.0, -5.1, -7.6, -2.2, -6.1, -3.6, -6.1, -6.0, -9.7, -7.3, -8.7, 11.2, 7.9, 8.9, 8.1, -12.1, -17.4, -13.2, -16.8, -6.3, -9.5, -7.5, -8.8, -12.7, -15.2, -13.1, -14.2, 0.0, -3.1, -1.8, -3.1, -50.0, -10.7, -50.0, -9.3, -4.6, -8.9, -5.9, -8.3, -50.0, -12.8, -9.6, -10.6, 9.5, -3.2, 8.2, -2.8,],\n            [-13.5, -11.3, -14.0, -12.9, -9.3, -7.1, -8.2, -8.9, -12.7, -10.3, -11.9, -12.3, -0.4, 1.4, -0.6, -0.4, -7.2, -4.7, -6.9, -5.7, -7.4, -3.7, -6.5, -6.0, -9.1, -5.3, -8.8, -7.4, 7.9, 11.9, 7.8, 9.3, -15.2, -15.1, -14.6, -17.2, -9.7, -7.3, -8.2, -9.2, -15.0, -12.8, -14.1, -14.9, -2.8, 0.2, -2.9, -2.3, -50.0, -7.4, -50.0, -8.7, -9.4, -7.3, -9.4, -9.4, -50.0, -8.0, -11.9, -9.7, 6.5, 0.5, 6.4, -1.8,],\n            [-13.0, -14.4, -11.9, -13.9, -7.7, -9.0, -6.7, -8.8, -11.7, -12.7, -10.7, -13.1, -0.2, -1.6, 1.7, -1.6, -5.7, -8.7, -4.8, -8.1, -5.3, -7.5, -3.3, -7.5, -9.0, -9.2, -6.5, -9.4, 8.9, 7.8, 10.1, 7.8, -14.3, -17.7, -13.0, -17.8, -8.0, -9.4, -6.2, -9.2, -15.1, -14.9, -12.2, -14.9, -1.8, -3.2, -0.8, -3.2, -50.0, -10.8, -50.0, -9.9, -7.3, -9.6, -6.4, -9.4, -50.0, -12.3, -8.7, -11.3, 7.6, -3.3, 8.7, -2.9,],\n            [-12.5, -12.9, -13.4, -10.0, -8.5, -9.0, -7.8, -6.8, -11.8, -11.3, -12.4, -10.1, -0.3, -0.5, -0.5, 1.2, -6.1, -6.0, -6.7, -3.5, -5.9, -5.6, -5.5, -2.9, -9.5, -7.8, -8.5, -5.3, 8.1, 9.3, 7.8, 11.8, -14.4, -15.7, -14.5, -14.6, -8.4, -8.7, -7.9, -7.1, -14.3, -13.8, -13.8, -12.3, -2.4, -2.2, -2.9, -0.2, -50.0, -8.9, -50.0, -7.1, -8.1, -9.1, -8.7, -6.6, -50.0, -9.7, -11.4, -8.0, 6.8, -1.8, 6.9, 0.4,],\n            [-2.6, -6.3, -4.9, -4.9, -6.4, -9.4, -7.9, -8.6, -6.3, -7.4, -8.2, -6.6, -9.7, -14.9, -11.1, -13.9, -0.2, -8.2, -2.3, -6.8, -8.9, -11.1, -9.4, -10.9, -8.9, -11.7, -10.0, -9.9, -12.1, -15.2, -14.3, -14.4, 11.1, 2.9, 9.2, 3.5, -3.0, -7.0, -4.3, -6.2, -2.2, -6.6, -3.7, -5.7, -6.0, -10.4, -8.6, -9.5, -50.0, -15.5, -50.0, -13.9, -7.8, -10.5, -8.7, -9.6, -50.0, -16.5, -17.1, -14.8, -12.1, -18.9, -12.4, -17.4,],\n            [-8.5, 0.8, -9.1, -0.7, -10.1, -7.5, -9.1, -8.2, -11.4, -3.1, -11.8, -4.4, -15.5, -14.3, -14.8, -15.4, -6.6, -4.7, -7.1, -5.8, -12.5, -10.3, -11.0, -12.2, -13.4, -10.6, -12.5, -11.8, -17.4, -15.1, -17.7, -15.7, 2.9, 12.5, 3.1, 10.3, -7.7, -5.2, -6.2, -7.3, -5.4, -1.7, -5.4, -3.7, -11.6, -9.1, -13.0, -10.8, -50.0, -10.3, -50.0, -11.2, -9.8, -8.3, -9.1, -9.7, -50.0, -12.7, -20.6, -13.1, -16.0, -16.6, -17.0, -18.4,],\n            [-5.0, -6.4, -3.4, -5.6, -7.5, -9.3, -7.1, -8.8, -8.8, -7.4, -6.9, -6.9, -11.4, -15.2, -9.7, -13.7, -1.8, -7.8, -0.8, -7.2, -9.4, -10.6, -8.7, -11.2, -10.0, -10.1, -7.8, -10.5, -13.2, -14.6, -13.0, -14.5, 9.2, 3.1, 10.8, 3.0, -4.5, -6.7, -2.7, -6.7, -5.0, -6.8, -2.5, -6.7, -7.5, -10.4, -7.4, -10.2, -50.0, -15.5, -50.0, -14.2, -8.9, -10.2, -8.5, -10.0, -50.0, -15.6, -15.3, -15.5, -13.0, -18.8, -12.6, -18.3,],\n            [-8.1, -1.1, -8.9, 1.4, -8.6, -9.2, -9.4, -6.8, -11.0, -4.9, -11.5, -2.6, -13.7, -16.8, -14.5, -12.6, -6.0, -6.7, -7.2, -4.2, -11.6, -11.9, -11.2, -10.9, -13.1, -12.2, -12.5, -10.3, -16.8, -17.2, -17.8, -14.6, 3.5, 10.3, 3.0, 12.4, -6.9, -7.5, -6.8, -4.9, -5.4, -4.2, -5.4, -1.3, -10.8, -11.5, -12.4, -8.2, -50.0, -12.5, -50.0, -8.9, -9.2, -10.0, -9.4, -7.9, -50.0, -14.3, -19.9, -11.4, -16.2, -19.2, -15.8, -15.9,],\n            [-6.3, -7.4, -8.0, -6.0, 2.8, -0.7, 0.9, 0.3, -7.5, -4.4, -8.4, -3.6, -3.0, -7.7, -3.7, -6.2, -5.3, -10.0, -7.2, -8.9, -0.7, -3.2, -1.8, -2.4, -9.9, -11.7, -9.6, -9.9, -6.3, -9.7, -8.0, -8.4, -3.0, -7.7, -4.5, -6.9, 11.3, 8.2, 9.4, 9.1, -1.0, -3.3, -1.8, -2.4, 1.5, -2.1, -0.4, -1.1, -50.0, -14.5, -50.0, -12.9, 2.8, -0.0, 1.4, 0.7, -50.0, -8.7, -14.1, -7.5, -5.2, -12.5, -6.5, -10.7,],\n            [-9.9, -5.0, -10.0, -6.5, -0.7, 2.6, -0.2, 0.1, -10.3, -1.8, -9.9, -3.8, -6.4, -4.7, -5.8, -6.7, -8.2, -8.3, -8.0, -10.2, -3.9, -1.0, -2.7, -2.9, -11.7, -8.8, -11.0, -10.1, -9.5, -7.3, -9.4, -8.7, -7.0, -5.2, -6.7, -7.5, 8.2, 11.6, 8.8, 9.1, -3.7, -0.9, -3.1, -3.0, -2.0, 1.2, -2.3, -1.3, -50.0, -12.1, -50.0, -12.6, 0.1, 2.5, 0.7, 0.6, -50.0, -6.0, -14.8, -7.5, -8.6, -9.4, -8.5, -11.2,],\n            [-7.5, -6.2, -7.2, -6.2, 0.9, 0.2, 2.5, 0.2, -8.3, -3.5, -7.1, -3.7, -4.1, -6.3, -3.1, -6.3, -5.7, -8.9, -5.6, -8.4, -2.3, -2.2, 0.7, -2.7, -9.8, -8.9, -7.1, -9.3, -7.5, -8.2, -6.2, -7.9, -4.3, -6.2, -2.7, -6.8, 9.4, 8.8, 12.1, 8.8, -2.1, -2.0, 0.1, -2.4, 0.0, -1.1, 1.3, -0.8, -50.0, -11.9, -50.0, -12.6, 1.2, 0.7, 3.5, 0.8, -50.0, -7.1, -12.4, -7.1, -7.1, -10.6, -5.0, -10.6,],\n            [-9.0, -6.5, -10.0, -4.8, 0.1, -0.1, -0.3, 2.4, -9.9, -3.5, -10.6, -2.0, -5.4, -6.6, -5.6, -4.4, -7.6, -9.9, -8.5, -8.2, -3.0, -2.8, -3.0, -0.9, -12.8, -11.8, -10.7, -9.3, -8.8, -9.2, -9.2, -7.1, -6.2, -7.3, -6.7, -4.9, 9.1, 9.1, 8.8, 11.2, -3.1, -3.0, -3.3, -1.0, -1.1, -1.4, -2.0, 1.2, -50.0, -12.8, -50.0, -10.9, 0.8, 0.5, 0.7, 2.3, -50.0, -7.5, -16.5, -5.9, -7.8, -11.1, -7.8, -9.0,],\n            [-7.1, -5.6, -9.1, -5.1, -5.9, -8.0, -6.9, -7.4, -1.9, -1.6, -4.7, -1.4, -9.6, -14.6, -10.4, -13.0, -6.7, -10.8, -9.4, -9.9, -9.1, -10.5, -9.1, -10.4, -5.9, -9.3, -7.9, -8.1, -12.7, -15.0, -15.1, -14.3, -2.2, -5.4, -5.0, -5.4, -1.0, -3.7, -2.1, -3.1, 12.8, 9.6, 11.1, 10.1, -4.8, -9.4, -8.1, -8.6, -50.0, -18.3, -50.0, -15.4, -5.9, -7.5, -6.6, -7.1, -50.0, -9.5, -11.7, -8.4, -12.5, -17.3, -13.2, -15.4,],\n            [-10.2, -1.6, -10.2, -3.2, -7.3, -4.6, -6.5, -6.2, -6.7, 3.0, -6.8, 0.8, -11.9, -10.8, -11.5, -12.2, -9.6, -7.1, -9.7, -8.4, -9.8, -8.1, -8.2, -9.7, -9.4, -5.0, -8.0, -7.2, -15.2, -12.8, -14.9, -13.8, -6.6, -1.7, -6.8, -4.2, -3.3, -0.9, -2.0, -3.0, 9.6, 12.8, 9.7, 10.5, -8.5, -5.6, -9.3, -7.8, -50.0, -12.7, -50.0, -13.8, -6.8, -5.1, -5.3, -6.5, -50.0, -4.1, -12.8, -5.8, -14.4, -14.1, -13.6, -14.8,],\n            [-8.2, -4.7, -7.1, -5.0, -6.5, -7.7, -5.5, -6.9, -4.1, -0.8, -1.2, -0.9, -10.5, -12.8, -8.2, -13.0, -7.5, -10.0, -7.2, -9.7, -9.2, -9.4, -7.3, -10.2, -6.5, -8.0, -4.3, -8.0, -13.1, -14.1, -12.2, -13.8, -3.7, -5.4, -2.5, -5.4, -1.8, -3.1, 0.1, -3.3, 11.1, 9.7, 12.9, 9.8, -5.5, -8.1, -5.4, -7.8, -50.0, -16.4, -50.0, -14.8, -5.9, -7.1, -4.7, -7.2, -50.0, -8.4, -6.4, -8.1, -12.1, -16.2, -10.3, -15.8,],\n            [-9.2, -3.0, -9.9, -1.2, -6.5, -6.1, -6.8, -4.4, -6.4, 0.9, -7.0, 3.0, -11.8, -13.0, -11.1, -10.4, -9.3, -9.2, -9.8, -6.4, -9.5, -9.5, -9.4, -8.0, -8.1, -8.2, -9.4, -5.2, -14.2, -14.9, -14.9, -12.3, -5.7, -3.7, -6.7, -1.3, -2.4, -3.0, -2.4, -1.0, 10.1, 10.5, 9.8, 13.1, -7.2, -8.0, -8.4, -5.0, -50.0, -14.7, -50.0, -11.4, -6.3, -6.7, -5.8, -5.0, -50.0, -6.1, -13.4, -3.6, -13.0, -16.0, -13.7, -13.3,],\n            [-8.2, -10.8, -10.1, -9.8, -0.4, -3.6, -1.7, -2.9, -7.9, -8.4, -9.8, -8.2, 6.2, 2.3, 0.6, 2.9, -7.1, -11.6, -9.2, -11.1, -5.6, -8.3, -7.2, -8.4, -10.7, -13.6, -11.4, -12.0, 0.0, -2.8, -1.8, -2.4, -6.0, -11.6, -7.5, -10.8, 1.5, -2.0, 0.0, -1.1, -4.8, -8.5, -5.5, -7.2, 11.9, 8.6, 10.0, 9.0, -50.0, -14.2, -50.0, -12.8, -4.1, -7.4, -5.7, -6.5, -50.0, -11.0, -14.3, -9.9, 1.4, -7.9, -0.7, -6.8,],\n            [-12.5, -8.7, -13.0, -9.9, -3.9, -1.0, -3.4, -2.9, -12.5, -6.3, -11.6, -8.1, 3.3, 6.0, -1.6, 3.8, -10.5, -10.1, -11.1, -11.1, -9.0, -6.9, -8.2, -9.1, -14.0, -11.4, -12.3, -12.1, -3.1, 0.2, -3.2, -2.2, -10.4, -9.1, -10.4, -11.5, -2.1, 1.2, -1.1, -1.4, -9.4, -5.6, -8.1, -8.0, 8.6, 12.4, 8.6, 9.5, -50.0, -10.4, -50.0, -11.6, -7.2, -5.7, -7.5, -7.4, -50.0, -8.6, -15.7, -9.7, -3.0, -3.7, -3.2, -5.9,],\n            [-11.1, -11.9, -10.6, -11.3, -2.6, -4.1, -1.7, -3.5, -11.4, -9.6, -9.6, -9.2, 3.7, 2.5, 1.6, 2.5, -9.6, -12.3, -9.5, -11.8, -8.0, -9.4, -7.2, -9.3, -13.3, -12.8, -10.9, -12.6, -1.8, -2.9, -0.8, -2.9, -8.6, -13.0, -7.4, -12.4, -0.4, -2.3, 1.3, -2.0, -8.1, -9.3, -5.4, -8.4, 10.0, 8.6, 11.4, 8.9, -50.0, -13.7, -50.0, -13.2, -6.1, -8.1, -5.5, -7.5, -50.0, -11.3, -12.7, -10.1, -1.6, -7.8, -0.3, -7.2,],\n            [-11.4, -10.0, -12.7, -8.2, -3.1, -3.1, -2.8, -0.9, -11.7, -7.3, -12.1, -5.8, 3.6, 3.6, -1.1, 5.8, -9.4, -11.9, -10.7, -10.0, -8.5, -8.8, -8.2, -6.5, -13.9, -13.0, -12.9, -10.9, -3.1, -2.3, -3.2, -0.2, -9.5, -10.8, -10.2, -8.2, -1.1, -1.3, -0.8, 1.2, -8.6, -7.8, -7.8, -5.0, 9.0, 9.5, 8.9, 12.0, -50.0, -13.0, -50.0, -10.3, -6.2, -7.0, -7.1, -5.4, -50.0, -9.3, -14.0, -8.2, -2.4, -6.3, -2.4, -3.9,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 33.3, -50.0, 30.6, -50.0, -50.0, -50.0, -50.0, -50.0, 29.2, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-14.8, -6.2, -14.9, -7.6, -14.6, -11.6, -13.0, -12.2, -14.0, -9.0, -14.1, -10.1, -13.6, -11.2, -12.4, -12.5, -8.1, 2.2, -8.7, 0.4, -15.3, -11.3, -13.2, -13.3, -11.9, -7.2, -11.8, -7.9, -10.7, -7.4, -10.8, -8.9, -15.5, -10.3, -15.5, -12.5, -14.5, -12.1, -11.9, -12.8, -18.3, -12.7, -16.4, -14.7, -14.2, -10.4, -13.7, -13.0, -50.0, 15.1, -50.0, 13.3, -9.9, -5.8, -8.9, -7.6, -50.0, -1.5, -7.8, -3.0, -8.1, 3.6, -9.4, 2.0,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 30.6, -50.0, 35.2, -50.0, -50.0, -50.0, -50.0, -50.0, 28.5, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-13.8, -7.5, -14.4, -5.1, -12.6, -11.6, -11.5, -10.1, -13.2, -9.7, -12.7, -7.4, -11.9, -11.9, -11.5, -10.2, -7.2, 0.9, -8.6, 2.6, -14.1, -12.8, -13.3, -11.0, -9.7, -8.4, -11.2, -5.3, -9.3, -8.7, -9.9, -7.1, -13.9, -11.2, -14.2, -8.9, -12.9, -12.6, -12.6, -10.9, -15.4, -13.8, -14.8, -11.4, -12.8, -11.6, -13.2, -10.3, -50.0, 13.3, -50.0, 15.2, -8.6, -7.0, -8.4, -4.8, -50.0, -3.0, -7.1, -0.1, -7.3, 2.3, -8.7, 3.9,],\n            [-7.3, -6.3, -8.9, -5.5, 2.9, 0.2, 1.5, 1.2, -8.5, -2.3, -10.0, -1.7, -5.6, -10.5, -5.5, -9.6, -4.3, -7.9, -6.1, -7.2, 2.3, -0.6, 0.5, -0.0, -8.9, -11.0, -9.4, -9.3, -4.6, -9.4, -7.3, -8.1, -7.8, -9.8, -8.9, -9.2, 2.8, 0.1, 1.2, 0.8, -5.9, -6.8, -5.9, -6.3, -4.1, -7.2, -6.1, -6.2, -50.0, -9.9, -50.0, -8.6, 12.5, 9.4, 11.0, 9.8, -50.0, -4.8, -8.7, -3.5, -0.0, -8.6, -2.1, -7.0,],\n            [-10.1, -4.3, -10.8, -5.8, 0.1, 2.3, 0.7, 0.6, -10.4, -0.2, -10.8, -2.0, -8.8, -9.2, -8.0, -9.7, -6.8, -5.9, -7.5, -7.0, -1.4, 2.2, -0.9, -0.4, -10.8, -7.7, -9.8, -8.8, -8.9, -7.3, -9.6, -9.1, -10.5, -8.3, -10.2, -10.0, -0.0, 2.5, 0.7, 0.5, -7.5, -5.1, -7.1, -6.7, -7.4, -5.7, -8.1, -7.0, -50.0, -5.8, -50.0, -7.0, 9.4, 12.7, 10.1, 10.0, -50.0, -1.5, -11.3, -3.3, -5.4, -4.2, -5.6, -6.8,],\n            [-8.4, -6.2, -8.9, -5.9, 1.8, 0.4, 2.7, 1.2, -9.1, -1.2, -8.8, -1.5, -7.2, -10.5, -4.9, -9.6, -5.7, -7.4, -5.5, -7.4, 0.3, -0.3, 1.8, -0.4, -9.0, -8.7, -7.3, -8.9, -5.9, -9.4, -6.4, -8.7, -8.7, -9.1, -8.5, -9.4, 1.4, 0.7, 3.5, 0.7, -6.6, -5.3, -4.7, -5.8, -5.7, -7.5, -5.5, -7.1, -50.0, -8.9, -50.0, -8.4, 11.0, 10.1, 13.2, 10.2, -50.0, -4.2, -6.1, -3.6, -3.5, -8.4, -1.2, -6.8,],\n            [-9.1, -5.4, -10.4, -4.2, 1.0, 0.5, 0.6, 2.4, -9.6, -1.5, -10.5, -0.5, -8.9, -10.5, -7.2, -8.3, -6.3, -6.9, -7.7, -5.1, -0.5, 0.1, -0.6, 2.3, -11.1, -10.2, -10.5, -7.0, -8.3, -9.4, -9.4, -6.6, -9.6, -9.7, -10.0, -7.9, 0.7, 0.6, 0.8, 2.3, -7.1, -6.5, -7.2, -5.0, -6.5, -7.4, -7.5, -5.4, -50.0, -7.6, -50.0, -4.8, 9.8, 10.0, 10.2, 12.1, -50.0, -3.1, -11.4, -0.7, -4.6, -6.6, -4.9, -3.9,],\n            [-50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, 29.2, -50.0, 28.5, -50.0, -50.0, -50.0, -50.0, -50.0, 33.3, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0, -50.0,],\n            [-13.0, -7.0, -13.1, -8.3, -9.2, -6.5, -8.9, -7.6, -9.8, -0.7, -9.2, -2.1, -12.4, -10.3, -12.4, -11.6, -9.5, -4.6, -10.4, -5.7, -13.6, -10.1, -11.4, -10.9, -6.5, -1.5, -6.2, -3.6, -12.8, -8.0, -12.3, -9.7, -16.5, -12.7, -15.6, -14.3, -8.7, -6.0, -7.1, -7.5, -9.5, -4.1, -8.4, -6.1, -11.0, -8.6, -11.3, -9.3, -50.0, -1.5, -50.0, -3.0, -4.8, -1.5, -4.2, -3.1, -50.0, 16.4, -5.1, 14.2, -10.1, -4.3, -10.0, -5.4,],\n            [-13.5, -16.3, -11.8, -15.5, -13.6, -15.3, -12.1, -16.1, -7.7, -13.3, -4.2, -12.6, -14.1, -16.2, -10.3, -15.4, -9.6, -11.6, -7.7, -9.8, -14.3, -17.5, -11.1, -17.2, -5.3, -7.1, -2.2, -7.1, -9.6, -11.9, -8.7, -11.4, -17.1, -20.6, -15.3, -19.9, -14.1, -14.8, -12.4, -16.5, -11.7, -12.8, -6.4, -13.4, -14.3, -15.7, -12.7, -14.0, -50.0, -7.8, -50.0, -7.1, -8.7, -11.3, -6.1, -11.4, -50.0, -5.1, 18.6, -4.8, -8.6, -8.1, -4.5, -7.3,],\n            [-12.4, -8.2, -13.0, -5.9, -7.7, -7.8, -7.4, -5.8, -8.8, -2.0, -9.3, -0.4, -11.6, -11.3, -11.6, -9.4, -8.9, -5.6, -10.0, -3.2, -11.5, -11.1, -10.5, -8.3, -6.3, -4.9, -6.7, -1.2, -10.6, -9.7, -11.3, -8.0, -14.8, -13.1, -15.5, -11.4, -7.5, -7.5, -7.1, -5.9, -8.4, -5.8, -8.1, -3.6, -9.9, -9.7, -10.1, -8.2, -50.0, -3.0, -50.0, -0.1, -3.5, -3.3, -3.6, -0.7, -50.0, 14.2, -4.8, 16.4, -8.9, -5.8, -8.9, -3.4,],\n            [-10.7, -13.2, -12.4, -11.6, -4.8, -8.4, -6.1, -7.2, -10.3, -11.8, -11.0, -11.6, 2.8, -1.7, 0.7, -1.1, -6.2, -9.7, -7.3, -8.4, -5.3, -8.6, -6.1, -8.3, -10.5, -11.2, -9.8, -10.2, 9.5, 6.5, 7.6, 6.8, -12.1, -16.0, -13.0, -16.2, -5.2, -8.6, -7.1, -7.8, -12.5, -14.4, -12.1, -13.0, 1.4, -3.0, -1.6, -2.4, -50.0, -8.1, -50.0, -7.3, -0.0, -5.4, -3.5, -4.6, -50.0, -10.1, -8.6, -8.9, 13.2, -0.9, 9.7, -0.3,],\n            [-18.1, -12.3, -19.4, -14.2, -12.2, -9.5, -11.5, -10.8, -17.9, -11.9, -18.2, -13.9, -6.4, -3.7, -6.8, -5.4, -12.8, -5.9, -13.6, -7.4, -13.7, -9.9, -12.9, -12.7, -16.2, -11.6, -15.2, -12.7, -3.2, 0.5, -3.3, -1.8, -18.9, -16.6, -18.8, -19.2, -12.5, -9.4, -10.6, -11.1, -17.3, -14.1, -16.2, -16.0, -7.9, -3.7, -7.8, -6.3, -50.0, 3.6, -50.0, 2.3, -8.6, -4.2, -8.4, -6.6, -50.0, -4.3, -8.1, -5.8, -0.9, 14.2, -1.8, 11.6,],\n            [-11.8, -13.1, -11.5, -12.0, -6.2, -8.3, -5.4, -7.4, -11.8, -12.3, -11.1, -11.4, 0.5, -2.5, 2.4, -1.6, -6.8, -9.9, -6.4, -8.9, -6.1, -8.6, -5.0, -7.8, -10.4, -11.6, -8.5, -9.9, 8.2, 6.4, 8.7, 6.9, -12.4, -17.0, -12.6, -15.8, -6.5, -8.5, -5.0, -7.8, -13.2, -13.6, -10.3, -13.7, -0.7, -3.2, -0.3, -2.4, -50.0, -9.4, -50.0, -8.7, -2.1, -5.6, -1.2, -4.9, -50.0, -10.0, -4.5, -8.9, 9.7, -1.8, 11.3, -0.9,],\n            [-17.2, -13.3, -17.8, -11.6, -11.0, -11.1, -10.4, -9.0, -16.0, -12.7, -16.1, -11.1, -5.2, -5.4, -6.0, -3.3, -11.8, -7.3, -13.3, -5.9, -12.7, -11.9, -11.9, -9.3, -15.2, -15.0, -14.9, -11.0, -2.8, -1.8, -2.9, 0.4, -17.4, -18.4, -18.3, -15.9, -10.7, -11.2, -10.6, -9.0, -15.4, -14.8, -15.8, -13.3, -6.8, -5.9, -7.2, -3.9, -50.0, 2.0, -50.0, 3.9, -7.0, -6.8, -6.8, -3.9, -50.0, -5.4, -7.3, -3.4, -0.3, 11.6, -0.9, 14.1,]];\n        this.TRANS = [[5, 0, 0, 4],\n            [0, 5, 4, 0],\n            [0, 4, 5, 0],\n            [4, 0, 0, 5]];\n        this.CONBLO = { 'A': 0, 'R': 1, 'N': 2, 'D': 3, 'C': 4, 'Q': 5, 'E': 6, 'G': 7, 'H': 8, 'I': 9, 'L': 10, 'K': 11, 'M': 12, 'F': 13, 'P': 14, 'S': 15, 'T': 16, 'W': 17, 'Y': 18, 'V': 19, 'B': 20, 'Z': 21, 'X': 22, '*': 23 };\n        this.CONNUCL = { 'A': 0, 'T': 1, 'C': 2, 'G': 3 };\n        this.CONSCHN = { 'AAA': 0, 'AAC': 1, 'AAG': 2, 'AAT': 3, 'ACA': 4, 'ACC': 5, 'ACG': 6, 'ACT': 7, 'AGA': 8, 'AGC': 9, 'AGG': 10, 'AGT': 11, 'ATA': 12, 'ATC': 13, 'ATG': 14, 'ATT': 15, 'CAA': 16, 'CAC': 17, 'CAG': 18, 'CAT': 19, 'CCA': 20, 'CCC': 21, 'CCG': 22, 'CCT': 23, 'CGA': 24, 'CGC': 25, 'CGG': 26, 'CGT': 27, 'CTA': 28, 'CTC': 29, 'CTG': 30, 'CTT': 31, 'GAA': 32, 'GAC': 33, 'GAG': 34, 'GAT': 35, 'GCA': 36, 'GCC': 37, 'GCG': 38, 'GCT': 39, 'GGA': 40, 'GGC': 41, 'GGG': 42, 'GGT': 43, 'GTA': 44, 'GTC': 45, 'GTG': 46, 'GTT': 47, 'TAA': 48, 'TAC': 49, 'TAG': 50, 'TAT': 51, 'TCA': 52, 'TCC': 53, 'TCG': 54, 'TCT': 55, 'TGA': 56, 'TGC': 57, 'TGG': 58, 'TGT': 59, 'TTA': 60, 'TTC': 61, 'TTG': 62, 'TTT': 63 };\n        this.setMethod(alignGrid_);\n        this.setSequences(seq1_, seq2_);\n        this.setGap(gap_);\n    }\n    fillNeedleman() {\n        for (let i = 0; i <= this.len1; i++) {\n            this.dpGrid[i][0].value = -i * this.gap;\n            this.dpGrid[i][0].parentI = i - 1;\n            this.dpGrid[i][0].parentJ = 0;\n        }\n        for (let i = 1; i <= this.len2; i++) {\n            this.dpGrid[0][i].value = -i * this.gap;\n            this.dpGrid[0][i].parentI = 0;\n            this.dpGrid[0][i].parentJ = i - 1;\n        }\n    }\n    fillOne(i, j) {\n        this.dpGrid[i][j].value = Math.max(this.dpGrid[i - 1][j - 1].value + this.scores[i - 1][j - 1], Math.max(this.dpGrid[i - 1][j].value - this.gap, this.dpGrid[i][j - 1].value - this.gap));\n        if (this.dpGrid[i][j].value == this.dpGrid[i - 1][j - 1].value + this.scores[i - 1][j - 1]) {\n            this.dpGrid[i][j].parentI = i - 1;\n            this.dpGrid[i][j].parentJ = j - 1;\n        }\n        else if (this.dpGrid[i][j].value == this.dpGrid[i - 1][j].value - this.gap) {\n            this.dpGrid[i][j].parentI = i - 1;\n            this.dpGrid[i][j].parentJ = j;\n        }\n        else {\n            this.dpGrid[i][j].parentI = i;\n            this.dpGrid[i][j].parentJ = j - 1;\n        }\n    }\n    gridFromName(path) {\n        switch (path) {\n            case 'BLOSUM45':\n                this.alignGrid = this.BLOSUM45;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM50':\n                this.alignGrid = this.BLOSUM50;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM62':\n                this.alignGrid = this.BLOSUM62;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM80':\n                this.alignGrid = this.BLOSUM80;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'BLOSUM90':\n                this.alignGrid = this.BLOSUM90;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM30':\n                this.alignGrid = this.PAM30;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM70':\n                this.alignGrid = this.PAM70;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'PAM250':\n                this.alignGrid = this.PAM250;\n                this.connections = this.CONBLO;\n                this.step = 1;\n                break;\n            case 'NUCLEOTIDES':\n                this.alignGrid = this.NUCLEOTIDES;\n                this.connections = this.CONNUCL;\n                this.step = 1;\n                break;\n            case 'TRANS':\n                this.alignGrid = this.TRANS;\n                this.connections = this.CONNUCL;\n                this.step = 1;\n                break;\n            case 'SCHNEIDER':\n                this.alignGrid = this.SCHNEIDER;\n                this.connections = this.CONSCHN;\n                this.step = 3;\n                break;\n        }\n        this.fillScores();\n    }\n    constructResult(i, j) {\n        let res1 = \"\", res2 = \"\", gapS = \"\";\n        let total = this.dpGrid[i][j].value;\n        for (let i = 0; i < this.step; i++) {\n            gapS += '|';\n        }\n        while (i > 0 || j > 0) {\n            if (this.dpGrid[i][j].parentI == i) {\n                res1 = gapS + res1;\n                res2 = this.seq2.substr((j - 1) * this.step, this.step) + res2;\n            }\n            else if (this.dpGrid[i][j].parentJ == j) {\n                res2 = gapS + res2;\n                res1 = this.seq1.substr((i - 1) * this.step, this.step) + res1;\n            }\n            else {\n                if (i - 1 >= 0)\n                    res1 = this.seq1.substr((i - 1) * this.step, this.step) + res1;\n                else\n                    res1 = gapS + res1;\n                if (j - 1 >= 0)\n                    res2 = this.seq2.substr((j - 1) * this.step, this.step) + res2;\n                else\n                    res2 = gapS + res2;\n            }\n            let tempI = i;\n            i = this.dpGrid[i][j].parentI;\n            j = this.dpGrid[tempI][j].parentJ;\n        }\n        return new Aligned(res1, res2, total);\n    }\n    fillScores() {\n        this.scores = [];\n        for (let i = 0; i < this.len1; i++) {\n            this.scores.push([]);\n            for (let j = 0; j < this.len2; j++) {\n                this.scores[i].push(this.alignGrid[this.connections[this.seq1.substr(i * this.step, this.step)]][this.connections[this.seq2.substr(j * this.step, this.step)]]);\n            }\n        }\n    }\n    typeRec() {\n        if (/[ATGC]*/.test(this.seq1) && /[ATGC]*/.test(this.seq2))\n            return \"NUCLEOTIDES\";\n        if (/[ARNDCQEGHILKMFPSTWYVBZX*]*/.test(this.seq1) && /[ARNDCQEGHILKMFPSTWYVBZX*]*/.test(this.seq2))\n            return \"BLOSUM62\";\n        throw Error(\"Can't recognize sequence type\");\n    }\n    needlemanWunch() {\n        this.dpGrid = [];\n        for (let i = 0; i < this.len1 + 1; i++) {\n            let row = [];\n            for (let j = 0; j < this.len2 + 1; j++)\n                row.push(new Cell());\n            this.dpGrid.push(row);\n        }\n        this.fillNeedleman();\n        for (let i = 1; i <= this.len1; i++) {\n            for (let j = 1; j <= this.len2; j++) {\n                this.fillOne(i, j);\n            }\n        }\n        return this.constructResult(this.len1, this.len2);\n    }\n    smithWaterman() {\n        this.dpGrid = [];\n        for (let i = 0; i < this.len1 + 1; i++) {\n            let row = [];\n            for (let j = 0; j < this.len2 + 1; j++)\n                row.push(new Cell());\n            this.dpGrid.push(row);\n        }\n        let bestVal = 0;\n        let bestI = 0, bestJ = 0;\n        for (let i = 1; i <= this.len1; i++) {\n            for (let j = 1; j <= this.len2; j++) {\n                this.fillOne(i, j);\n                if (this.dpGrid[i][j].value <= 0) {\n                    this.dpGrid[i][j].value = 0;\n                    this.dpGrid[i][j].parentI = 0;\n                    this.dpGrid[i][j].parentJ = 0;\n                }\n                else if (this.dpGrid[this.dpGrid[i][j].parentI][this.dpGrid[i][j].parentJ].value == 0) {\n                    this.dpGrid[i][j].parentI = 0;\n                    this.dpGrid[i][j].parentJ = 0;\n                }\n                if (this.dpGrid[i][j].value > bestVal) {\n                    bestVal = this.dpGrid[i][j].value;\n                    bestI = i;\n                    bestJ = j;\n                }\n            }\n        }\n        return this.constructResult(bestI, bestJ);\n    }\n    setSequences(seq1_, seq2_) {\n        this.seq1 = seq1_;\n        this.seq2 = seq2_;\n        this.len1 = this.seq1.length / this.step;\n        this.len2 = this.seq2.length / this.step;\n        this.fillScores();\n    }\n    setGap(gap_) {\n        this.gap = gap_;\n    }\n    setMethod(alignGrid_ = \"\") {\n        if (alignGrid_ == \"AUTO\") {\n            alignGrid_ = this.typeRec();\n        }\n        this.gridFromName(alignGrid_);\n    }\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nexport function getMacroMol(monomers) {\n    return __awaiter(this, void 0, void 0, function* () {\n        let result = [];\n        const moduleRdkit = yield grok.functions.call('Chem:getRdKitModule');\n        for (let i = 0; i < monomers.length; i++) {\n            for (let j = 0; j < monomers[i].length; j++) {\n                const mol = moduleRdkit.get_mol(monomers[i][j]['molfile']);\n                const a = mol.get_v3Kmolblock();\n                const indices = getIndices(monomers[i][j], a);\n                monomers[i][j]['indices'] = indices;\n                monomers[i][j]['molfile'] = yield rotateBackboneV3000(a, indices);\n                mol === null || mol === void 0 ? void 0 : mol.delete();\n            }\n            result.push(linkV3000(monomers[i]));\n        }\n        return result;\n    });\n}\nfunction getIndices(monomer, molV3000) {\n    const molfile = monomer[\"molfile\"];\n    let indexStart = molfile.indexOf('M  RGP', 0) + 8;\n    let indexEnd = molfile.indexOf('\\n', indexStart);\n    const indicesData = molfile.substring(indexStart, indexEnd).replaceAll('  ', ' ').replaceAll('  ', ' ');\n    let parsedData = indicesData.split(' ');\n    const remFirst = parsedData[2] == '1' ? parseInt(parsedData[1]) : parseInt(parsedData[3]);\n    const remLast = parsedData[2] == '2' ? parseInt(parsedData[1]) : parseInt(parsedData[3]);\n    const numbers = extractAtomsBondsNumbersV3000(molV3000);\n    let indexBonds = molV3000.indexOf('M  V30 BEGIN BOND'); // V3000 index for bonds\n    indexBonds = molV3000.indexOf('\\n', indexBonds);\n    indexStart = indexBonds;\n    indexEnd = indexBonds;\n    let first = 0;\n    let last = 0;\n    let remBondFirst = 0;\n    let remBondLast = 0;\n    for (let j = 0; j < numbers.nbond; j++) {\n        if (first == 0 || last == 0) {\n            indexStart = molV3000.indexOf('V30', indexStart) + 4;\n            indexEnd = molV3000.indexOf('\\n', indexStart);\n            const bondData = molV3000.substring(indexStart, indexEnd).replaceAll('  ', ' ').replaceAll('  ', ' ');\n            parsedData = bondData.split(' ');\n            if (parseInt(parsedData[2]) == remFirst) {\n                first = parseInt(parsedData[3]);\n                remBondFirst = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[3]) == remFirst) {\n                first = parseInt(parsedData[2]);\n                remBondFirst = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[2]) == remLast) {\n                last = parseInt(parsedData[3]);\n                remBondLast = parseInt(parsedData[0]);\n            }\n            else if (parseInt(parsedData[3]) == remLast) {\n                last = parseInt(parsedData[2]);\n                remBondLast = parseInt(parsedData[0]);\n            }\n        }\n    }\n    return { first, last, remFirst, remLast, remBondFirst, remBondLast };\n}\nfunction rotateBackboneV3000(molBlock, indices) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const coordinates = extractAtomDataV3000(molBlock);\n        const natom = coordinates.atomIndex.length;\n        const first = indices['first'];\n        const last = indices['last'];\n        const xCenter = (coordinates.x[last] + coordinates.x[first]) / 2;\n        const yCenter = (coordinates.y[last] + coordinates.y[first]) / 2;\n        //place to center\n        for (let i = 0; i < natom; i++) {\n            coordinates.x[i] -= xCenter;\n            coordinates.y[i] -= yCenter;\n        }\n        let angle = 0;\n        if (coordinates.x[first] == 0)\n            angle = coordinates.y[first] > coordinates.y[last] ? Math.PI / 2 : 3 * Math.PI / 2;\n        else if (coordinates.y[first] == 0)\n            angle = coordinates.x[first] > coordinates.x[last] ? Math.PI : 0;\n        else {\n            const derivative = coordinates.y[first] / coordinates.x[first];\n            if (coordinates.x[first] < coordinates.x[last])\n                angle = derivative > 0 ? Math.PI - Math.atan(derivative) : Math.atan(derivative);\n            else\n                angle = derivative > 0 ? Math.atan(derivative) : Math.PI - Math.atan(derivative);\n        }\n        const cos = Math.cos(angle);\n        const sin = Math.sin(angle);\n        for (let i = 0; i < natom; i++) {\n            const xAdd = coordinates.x[i];\n            coordinates.x[i] = xAdd * cos - coordinates.y[i] * sin;\n            coordinates.y[i] = xAdd * sin + coordinates.y[i] * cos;\n        }\n        //place to right\n        const xShift = coordinates.x[first];\n        for (let i = 0; i < natom; i++)\n            coordinates.x[i] -= xShift;\n        //rewrite molBlock\n        let index = molBlock.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n        index = molBlock.indexOf('\\n', index);\n        let indexEnd = index;\n        for (let i = 0; i < natom; i++) {\n            index = molBlock.indexOf('V30', index) + 4;\n            index = molBlock.indexOf(' ', index) + 1;\n            index = molBlock.indexOf(' ', index) + 1;\n            indexEnd = molBlock.indexOf(' ', index) + 1;\n            indexEnd = molBlock.indexOf(' ', indexEnd);\n            molBlock = molBlock.slice(0, index) +\n                coordinates.x[i] + ' ' + coordinates.y[i] +\n                molBlock.slice(indexEnd);\n            index = molBlock.indexOf('\\n', index) + 1;\n        }\n        return molBlock;\n    });\n}\nfunction extractAtomDataV3000(molBlock) {\n    const numbers = extractAtomsBondsNumbersV3000(molBlock);\n    let index = molBlock.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n    index = molBlock.indexOf('\\n', index);\n    let indexEnd = index;\n    const indexes = Array(numbers.natom);\n    const types = Array(numbers.natom);\n    const x = Array(numbers.natom);\n    const y = Array(numbers.natom);\n    for (let i = 0; i < numbers.natom; i++) {\n        index = molBlock.indexOf('V30', index) + 4;\n        indexEnd = molBlock.indexOf(' ', index);\n        indexes[i] = parseInt(molBlock.substring(index, indexEnd));\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        types[i] = molBlock.substring(index, indexEnd);\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        x[i] = parseFloat(molBlock.substring(index, indexEnd));\n        index = indexEnd + 1;\n        indexEnd = molBlock.indexOf(' ', index);\n        y[i] = parseFloat(molBlock.substring(index, indexEnd));\n        index = molBlock.indexOf('\\n', index) + 1;\n    }\n    return { atomIndex: indexes, atomType: types, x: x, y: y };\n}\nfunction linkV3000(monomers) {\n    let macroMolBlock = '\\nDatagrok macromolecule handler\\n\\n';\n    macroMolBlock += '  0  0  0  0  0  0            999 V3000\\n';\n    macroMolBlock += 'M  V30 BEGIN CTAB\\n';\n    let atomBlock = '';\n    let bondBlock = '';\n    let collectionBlock = '';\n    const collection = [];\n    let natom = 0;\n    let nbond = 0;\n    let xShift = 0;\n    for (let i = 0; i < monomers.length; i++) {\n        let molfile = monomers[i]['molfile'];\n        const first = monomers[i]['indices']['first'];\n        const last = monomers[i]['indices']['last'];\n        const remFirst = monomers[i]['indices']['remFirst'];\n        const remLast = monomers[i]['indices']['remLast'];\n        const remBondFirst = monomers[i]['indices']['remBondFirst'];\n        const remBondLast = monomers[i]['indices']['remBondLast'];\n        molfile = molfile.replaceAll('(-\\nM  V30 ', '(')\n            .replaceAll('-\\nM  V30 ', '').replaceAll(' )', ')');\n        const numbers = extractAtomsBondsNumbersV3000(molfile);\n        const coordinates = extractAtomDataV3000(molfile);\n        let indexAtoms = molfile.indexOf('M  V30 BEGIN ATOM'); // V3000 index for atoms coordinates\n        indexAtoms = molfile.indexOf('\\n', indexAtoms);\n        let index = indexAtoms;\n        let indexEnd = indexAtoms;\n        const totalShift = xShift - coordinates.x[first - 1];\n        for (let j = 0; j < numbers.natom; j++) {\n            if (coordinates.atomIndex[j] != remFirst && coordinates.atomIndex[j] != remLast) { //|| i == 0) {\n                //rewrite atom number\n                index = molfile.indexOf('V30', index) + 4;\n                indexEnd = molfile.indexOf(' ', index);\n                let atomNumber = parseInt(molfile.substring(index, indexEnd));\n                atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                    (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n                atomNumber += natom;\n                molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n                //rewrite coordinates\n                index = molfile.indexOf(' ', index) + 1;\n                index = molfile.indexOf(' ', index) + 1;\n                indexEnd = molfile.indexOf(' ', index);\n                let coordinate = Math.round(10000 * (parseFloat(molfile.substring(index, indexEnd)) + totalShift)) / 10000;\n                molfile = molfile.slice(0, index) + coordinate + molfile.slice(indexEnd);\n                index = molfile.indexOf(' ', index) + 1;\n                indexEnd = molfile.indexOf(' ', index);\n                coordinate = Math.round(10000 * (parseFloat(molfile.substring(index, indexEnd)))) / 10000;\n                molfile = molfile.slice(0, index) + coordinate + molfile.slice(indexEnd);\n                index = molfile.indexOf('\\n', index) + 1;\n            }\n            else {\n                index = molfile.indexOf('M  V30', index) - 1;\n                indexEnd = molfile.indexOf('\\n', index + 1);\n                molfile = molfile.slice(0, index) + molfile.slice(indexEnd);\n            }\n        }\n        const indexAtomsEnd = molfile.indexOf('M  V30 END ATOM');\n        atomBlock += molfile.substring(indexAtoms + 1, indexAtomsEnd);\n        let indexBonds = molfile.indexOf('M  V30 BEGIN BOND'); // V3000 index for bonds\n        indexBonds = molfile.indexOf('\\n', indexBonds);\n        index = indexBonds;\n        indexEnd = indexBonds;\n        let bondNumber = 0;\n        for (let j = 0; j < numbers.nbond; j++) {\n            //rewrite bond number\n            index = molfile.indexOf('V30', index) + 4;\n            indexEnd = molfile.indexOf(' ', index);\n            bondNumber = parseInt(molfile.substring(index, indexEnd));\n            if (bondNumber == remBondFirst || bondNumber == remBondLast) {\n                indexEnd = molfile.indexOf('\\n', index) + 1;\n                index -= 7;\n                molfile = molfile.slice(0, index) + molfile.slice(indexEnd);\n                continue;\n            }\n            bondNumber = (bondNumber > remBondFirst && bondNumber > remBondLast) ? bondNumber - 2 :\n                (bondNumber > remBondFirst || bondNumber > remBondLast) ? bondNumber - 1 : bondNumber;\n            bondNumber += nbond;\n            molfile = molfile.slice(0, index) + bondNumber + molfile.slice(indexEnd);\n            //rewrite atom pair in bond\n            index = molfile.indexOf(' ', index) + 1;\n            index = molfile.indexOf(' ', index) + 1;\n            indexEnd = molfile.indexOf(' ', index);\n            let atomNumber = parseInt(molfile.substring(index, indexEnd));\n            atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n            atomNumber += natom;\n            molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n            index = molfile.indexOf(' ', index) + 1;\n            indexEnd = Math.min(molfile.indexOf('\\n', index), molfile.indexOf(' ', index));\n            atomNumber = parseInt(molfile.substring(index, indexEnd));\n            atomNumber = (atomNumber > remFirst && atomNumber > remLast) ? atomNumber - 2 :\n                (atomNumber > remFirst || atomNumber > remLast) ? atomNumber - 1 : atomNumber;\n            atomNumber += natom;\n            molfile = molfile.slice(0, index) + atomNumber + molfile.slice(indexEnd);\n            index = molfile.indexOf('\\n', index) + 1;\n        }\n        const indexBondEnd = molfile.indexOf('M  V30 END BOND');\n        bondBlock += molfile.substring(indexBonds + 1, indexBondEnd);\n        //let indexCollection = molfile.indexOf('M  V30 MDLV30/STEABS ATOMS=('); // V3000 index for collections\n        // while (indexCollection != -1) {\n        //   indexCollection += 28;\n        //   const collectionEnd = molfile.indexOf(')', indexCollection);\n        //   const collectionEntries = molfile.substring(indexCollection, collectionEnd).split(' ').slice(1);\n        //   collectionEntries.forEach((e: string) => {\n        //     collection.push(parseInt(e) + natom);\n        //   });\n        //   indexCollection = collectionEnd;\n        //   indexCollection = molfile.indexOf('M  V30 MDLV30/STEABS ATOMS=(', indexCollection);\n        // }\n        natom += numbers.natom - 2;\n        nbond += numbers.nbond - 2;\n        xShift += coordinates.x[last] - coordinates.x[first] + 1;\n        if (i == monomers.length - 1) {\n            natom++;\n            const shift = xShift + 0.2;\n            atomBlock += 'M  V30 ' + natom + ' O ' + shift + ' 0 0.000000 0\\n';\n        }\n        nbond++;\n        if (i == monomers.length - 1) {\n            const rightTerminal = (last > remFirst && last > remLast) ? last + natom - (numbers.natom - 2) - 3 :\n                (last > remFirst || last > remLast) ? last + natom - (numbers.natom - 2) - 2 :\n                    last + natom - (numbers.natom - 2) - 1;\n            bondBlock += 'M  V30 ' + nbond + ' 1 ' + rightTerminal + ' ' + natom + '\\n';\n        }\n        else {\n            const rightTerminal = (last > remFirst && last > remLast) ? last + natom - (numbers.natom - 2) - 2 :\n                (last > remFirst || last > remLast) ? last + natom - (numbers.natom - 2) - 1 :\n                    last + natom - (numbers.natom - 2);\n            const next = monomers[i + 1]['indices'];\n            const nextFirst = next['first'];\n            const nextRemFirst = next['remFirst'];\n            const nextRemLast = next['remLast'];\n            const leftTerminal = (nextFirst > nextRemFirst && nextFirst > nextRemLast) ? nextFirst + natom - 2 :\n                (nextFirst > nextRemFirst || nextFirst > nextRemLast) ? nextFirst + natom - 1 :\n                    nextFirst + natom;\n            bondBlock += 'M  V30 ' + nbond + ' 1 ' + rightTerminal + ' ' + leftTerminal + '\\n';\n        }\n    }\n    // const entries = 4;\n    // const collNumber = Math.ceil(collection.length / entries);\n    // collectionBlock += 'M  V30 MDLV30/STEABS ATOMS=(' + collection.length + ' -\\n';\n    // for (let i = 0; i < collNumber; i++) {\n    //   collectionBlock += 'M  V30 ';\n    //   const entriesCurrent = i + 1 == collNumber ? collection.length - (collNumber - 1)*entries : entries;\n    //   for (let j = 0; j < entriesCurrent; j++) {\n    //     collectionBlock += (j + 1 == entriesCurrent) ?\n    //       (i == collNumber - 1 ? collection[entries*i + j] + ')\\n' : collection[entries*i + j] + ' -\\n') :\n    //       collection[entries*i + j] + ' ';\n    //   }\n    // }\n    //generate file\n    macroMolBlock += 'M  V30 COUNTS ' + natom + ' ' + nbond + ' 0 0 0\\n';\n    macroMolBlock += 'M  V30 BEGIN ATOM\\n';\n    macroMolBlock += atomBlock;\n    macroMolBlock += 'M  V30 END ATOM\\n';\n    macroMolBlock += 'M  V30 BEGIN BOND\\n';\n    macroMolBlock += bondBlock;\n    macroMolBlock += 'M  V30 END BOND\\n';\n    //macroMolBlock += 'M  V30 BEGIN COLLECTION\\n';\n    //macroMolBlock += collectionBlock;\n    //macroMolBlock += 'M  V30 END COLLECTION\\n';\n    macroMolBlock += 'M  V30 END CTAB\\n';\n    macroMolBlock += 'M  END\\n';\n    return macroMolBlock;\n}\nfunction extractAtomsBondsNumbersV3000(molBlock) {\n    molBlock = molBlock.replaceAll('\\r', ''); //equalize old and new sdf standards\n    let index = molBlock.indexOf('COUNTS') + 7; // V3000 index for atoms and bonds number\n    let indexEnd = molBlock.indexOf(' ', index);\n    const atomsNumber = parseInt(molBlock.substring(index, indexEnd));\n    index = indexEnd + 1;\n    indexEnd = molBlock.indexOf(' ', index);\n    const bondsNumber = parseInt(molBlock.substring(index, indexEnd));\n    return { natom: atomsNumber, nbond: bondsNumber };\n}\n","import * as C from './constants';\nimport * as DG from 'datagrok-api/dg';\nimport { AminoacidsPalettes } from '@datagrok-libraries/bio/src/aminoacids';\nimport { NucleotidesPalettes } from '@datagrok-libraries/bio/src/nucleotides';\nimport { UnknownSeqPalettes } from '@datagrok-libraries/bio/src/unknown';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport * as ui from 'datagrok-api/ui';\nimport { printLeftOrCentered, DrawStyle } from '@datagrok-libraries/bio/src/utils/cell-renderer';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nconst undefinedColor = 'rgb(100,100,100)';\nconst monomerToShortFunction = WebLogo.monomerToShort;\nconst gapRenderer = 5;\nfunction getPaletteByType(paletteType) {\n    switch (paletteType) {\n        case 'PT':\n            return AminoacidsPalettes.GrokGroups;\n        case 'NT':\n            return NucleotidesPalettes.Chromatogram;\n        case 'DNA':\n            return NucleotidesPalettes.Chromatogram;\n        case 'RNA':\n            return NucleotidesPalettes.Chromatogram;\n        // other\n        default:\n            return UnknownSeqPalettes.Color;\n    }\n}\nfunction getUpdatedWidth(grid, g, x, w) {\n    return grid ? Math.min(grid.canvas.width - x, w) : g.canvas.width - x;\n}\nexport function processSequence(subParts) {\n    const simplified = !subParts.some((amino, index) => amino.length > 1 &&\n        index != 0 &&\n        index != subParts.length - 1);\n    const text = [];\n    const gap = simplified ? '' : ' ';\n    subParts.forEach((amino, index) => {\n        if (index < subParts.length)\n            amino += `${amino ? '' : '-'}${gap}`;\n        text.push(amino);\n    });\n    return [text, simplified];\n}\nexport class MacromoleculeSequenceCellRenderer extends DG.GridCellRenderer {\n    get name() { return 'sequence'; }\n    get cellType() { return 'sequence'; }\n    get defaultHeight() { return 30; }\n    get defaultWidth() { return 230; }\n    onMouseMove(gridCell, e) {\n        var _a, _b, _c;\n        if (gridCell.cell.column.getTag(UnitsHandler.TAGS.aligned) !== 'SEQ.MSA') {\n            return;\n        }\n        const maxLengthWordsSum = gridCell.cell.column.temp['bio-sum-maxLengthWords'];\n        const maxIndex = gridCell.cell.column.temp['bio-maxIndex'];\n        const argsX = e.offsetX - gridCell.gridColumn.left + (gridCell.gridColumn.left - gridCell.bounds.x);\n        let left = 0;\n        let right = maxIndex;\n        let found = false;\n        maxLengthWordsSum[maxIndex + 1] = argsX + 1;\n        let mid = 0;\n        if (argsX > maxLengthWordsSum[0]) {\n            while (!found) {\n                mid = Math.floor((right + left) / 2);\n                if (argsX >= maxLengthWordsSum[mid] && argsX <= maxLengthWordsSum[mid + 1]) {\n                    left = mid;\n                    found = true;\n                }\n                else if (argsX < maxLengthWordsSum[mid]) {\n                    right = mid - 1;\n                }\n                else if (argsX > maxLengthWordsSum[mid + 1]) {\n                    left = mid + 1;\n                }\n                if (left == right) {\n                    found = true;\n                }\n            }\n        }\n        left = (argsX >= maxLengthWordsSum[left]) ? left + 1 : left;\n        const separator = (_a = gridCell.cell.column.getTag('separator')) !== null && _a !== void 0 ? _a : '';\n        const splitterFunc = WebLogo.getSplitter('separator', separator);\n        const subParts = splitterFunc(gridCell.cell.value);\n        ((((_c = (_b = subParts[left]) === null || _b === void 0 ? void 0 : _b.length) !== null && _c !== void 0 ? _c : 0) > 0)) ? ui.tooltip.show(ui.div(subParts[left]), e.x + 16, e.y + 16) : ui.tooltip.hide();\n    }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     * @memberof AlignedSequenceCellRenderer\n     */\n    render(g, x, y, w, h, gridCell, cellStyle) {\n        var _a;\n        const grid = gridCell.gridRow !== -1 ? gridCell.grid : null;\n        const cell = gridCell.cell;\n        const paletteType = gridCell.cell.column.getTag(C.TAGS.ALPHABET);\n        const minDistanceRenderer = 50;\n        w = getUpdatedWidth(grid, g, x, w);\n        g.save();\n        g.beginPath();\n        g.rect(x, y, w, h);\n        g.clip();\n        g.font = '12px monospace';\n        g.textBaseline = 'top';\n        //TODO: can this be replaced/merged with splitSequence?\n        const units = gridCell.cell.column.getTag(DG.TAGS.UNITS);\n        const palette = getPaletteByType(paletteType);\n        const separator = (_a = gridCell.cell.column.getTag('separator')) !== null && _a !== void 0 ? _a : '';\n        const splitLimit = gridCell.bounds.width / 5;\n        const splitterFunc = WebLogo.getSplitter(units, separator, gridCell.bounds.width / 5);\n        const maxLengthOfMonomer = 8;\n        let maxLengthWords = {};\n        if (gridCell.cell.column.getTag('.calculatedCellRender') !== splitLimit.toString()) {\n            let samples = 0;\n            while (samples < Math.min(gridCell.cell.column.length, 100)) {\n                let column = gridCell.cell.column.get(samples);\n                let subParts = splitterFunc(column);\n                subParts.forEach((amino, index) => {\n                    var _a, _b;\n                    let textSize = monomerToShortFunction(amino, maxLengthOfMonomer).length * 7 + gapRenderer;\n                    if (textSize > ((_a = maxLengthWords[index]) !== null && _a !== void 0 ? _a : 0)) {\n                        maxLengthWords[index] = textSize;\n                    }\n                    if (index > ((_b = maxLengthWords['bio-maxIndex']) !== null && _b !== void 0 ? _b : 0)) {\n                        maxLengthWords['bio-maxIndex'] = index;\n                    }\n                });\n                samples += 1;\n            }\n            let minLength = 3 * 7;\n            for (let i = 0; i <= maxLengthWords['bio-maxIndex']; i++) {\n                if (maxLengthWords[i] < minLength) {\n                    maxLengthWords[i] = minLength;\n                }\n            }\n            let maxLengthWordSum = {};\n            maxLengthWordSum[0] = maxLengthWords[0];\n            for (let i = 1; i <= maxLengthWords['bio-maxIndex']; i++) {\n                maxLengthWordSum[i] = maxLengthWordSum[i - 1] + maxLengthWords[i];\n            }\n            gridCell.cell.column.temp = {\n                'bio-sum-maxLengthWords': maxLengthWordSum,\n                'bio-maxIndex': maxLengthWords['bio-maxIndex'],\n                'bio-maxLengthWords': maxLengthWords\n            };\n            gridCell.cell.column.setTag('.calculatedCellRender', splitLimit.toString());\n        }\n        else {\n            maxLengthWords = gridCell.cell.column.temp['bio-maxLengthWords'];\n        }\n        const subParts = splitterFunc(cell.value);\n        let x1 = x;\n        let color = undefinedColor;\n        let drawStyle = DrawStyle.classic;\n        if (gridCell.cell.column.getTag('aligned').includes('MSA') && gridCell.cell.column.getTag('units') === 'separator') {\n            drawStyle = DrawStyle.MSA;\n        }\n        subParts.every((amino, index) => {\n            color = palette.get(amino);\n            g.fillStyle = undefinedColor;\n            let last = index === subParts.length - 1;\n            x1 = printLeftOrCentered(x1, y, w, h, g, monomerToShortFunction(amino, maxLengthOfMonomer), color, 0, true, 1.0, separator, last, drawStyle, maxLengthWords, index, gridCell);\n            return x1 - minDistanceRenderer - gridCell.gridColumn.left + (gridCell.gridColumn.left - gridCell.bounds.x) <= gridCell.bounds.width;\n        });\n        g.restore();\n        return;\n    }\n}\nexport class MonomerCellRenderer extends DG.GridCellRenderer {\n    get name() { return C.SEM_TYPES.MONOMER; }\n    get cellType() { return C.SEM_TYPES.MONOMER; }\n    get defaultHeight() { return 15; }\n    get defaultWidth() { return 30; }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     */\n    render(g, x, y, w, h, gridCell, _cellStyle) {\n        g.font = `12px monospace`;\n        g.textBaseline = 'middle';\n        g.textAlign = 'center';\n        const palette = getPaletteByType(gridCell.cell.column.getTag(C.TAGS.ALPHABET));\n        const s = gridCell.cell.value || '-';\n        const color = palette.get(s);\n        g.fillStyle = color;\n        g.fillText(s, x + (w / 2), y - (h / 2), w);\n    }\n}\nexport class MacromoleculeDifferenceCellRenderer extends DG.GridCellRenderer {\n    get name() { return 'MacromoleculeDifferenceCR'; }\n    get cellType() { return C.SEM_TYPES.MACROMOLECULE_DIFFERENCE; }\n    get defaultHeight() { return 30; }\n    get defaultWidth() { return 230; }\n    /**\n     * Cell renderer function.\n     *\n     * @param {CanvasRenderingContext2D} g Canvas rendering context.\n     * @param {number} x x coordinate on the canvas.\n     * @param {number} y y coordinate on the canvas.\n     * @param {number} w width of the cell.\n     * @param {number} h height of the cell.\n     * @param {DG.GridCell} gridCell Grid cell.\n     * @param {DG.GridCellStyle} cellStyle Cell style.\n     * @memberof AlignedSequenceDifferenceCellRenderer\n     */\n    render(g, x, y, w, h, gridCell, cellStyle) {\n        var _a;\n        const grid = gridCell.grid;\n        const cell = gridCell.cell;\n        w = getUpdatedWidth(grid, g, w, x);\n        g.save();\n        g.beginPath();\n        g.rect(x, y, w, h);\n        g.clip();\n        g.font = '12px monospace';\n        g.textBaseline = 'top';\n        const s = (_a = cell.value) !== null && _a !== void 0 ? _a : '';\n        //TODO: can this be replaced/merged with splitSequence?\n        const [s1, s2] = s.split('#');\n        const separator = gridCell.tableColumn.tags[C.TAGS.SEPARATOR];\n        const units = gridCell.tableColumn.tags[DG.TAGS.UNITS];\n        const splitter = WebLogo.getSplitter(units, separator);\n        const subParts1 = splitter(s1);\n        const subParts2 = splitter(s2);\n        const [text] = processSequence(subParts1);\n        const textSize = g.measureText(text.join(''));\n        let updatedX = Math.max(x, x + (w - (textSize.width + subParts1.length * 4)) / 2);\n        // 28 is the height of the two substitutions on top of each other + space\n        const updatedY = Math.max(y, y + (h - 28) / 2);\n        let palette = UnknownSeqPalettes.Color;\n        if (units != 'HELM')\n            palette = getPaletteByType(units.substring(units.length - 2));\n        const vShift = 7;\n        for (let i = 0; i < subParts1.length; i++) {\n            const amino1 = subParts1[i];\n            const amino2 = subParts2[i];\n            const color1 = palette.get(amino1);\n            if (amino1 != amino2) {\n                const color2 = palette.get(amino2);\n                const subX0 = printLeftOrCentered(updatedX, updatedY - vShift, w, h, g, amino1, color1, 0, true);\n                const subX1 = printLeftOrCentered(updatedX, updatedY + vShift, w, h, g, amino2, color2, 0, true);\n                updatedX = Math.max(subX1, subX0);\n            }\n            else\n                updatedX = printLeftOrCentered(updatedX, updatedY, w, h, g, amino1, color1, 0, true, 0.5);\n            updatedX += 4;\n        }\n        g.restore();\n    }\n}\n","export var COLUMNS_NAMES;\n(function (COLUMNS_NAMES) {\n    COLUMNS_NAMES[\"SPLIT_COL\"] = \"~split\";\n    COLUMNS_NAMES[\"ACTIVITY\"] = \"~activity\";\n    COLUMNS_NAMES[\"ACTIVITY_SCALED\"] = \"activity_scaled\";\n    COLUMNS_NAMES[\"ALIGNED_SEQUENCE\"] = \"~aligned_sequence\";\n    COLUMNS_NAMES[\"AMINO_ACID_RESIDUE\"] = \"AAR\";\n    COLUMNS_NAMES[\"POSITION\"] = \"Pos\";\n    COLUMNS_NAMES[\"P_VALUE\"] = \"pValue\";\n    COLUMNS_NAMES[\"MEAN_DIFFERENCE\"] = \"Mean difference\";\n})(COLUMNS_NAMES || (COLUMNS_NAMES = {}));\nexport var TAGS;\n(function (TAGS) {\n    TAGS[\"AAR\"] = \"AAR\";\n    TAGS[\"POSITION\"] = \"Pos\";\n    TAGS[\"SEPARATOR\"] = \"separator\";\n    TAGS[\"SELECTION\"] = \"selection\";\n    TAGS[\"ALPHABET\"] = \"alphabet\";\n    TAGS[\"ALIGNED\"] = \"aligned\";\n})(TAGS || (TAGS = {}));\nexport var SEM_TYPES;\n(function (SEM_TYPES) {\n    SEM_TYPES[\"MONOMER\"] = \"Monomer\";\n    SEM_TYPES[\"MACROMOLECULE_DIFFERENCE\"] = \"MacromoleculeDifference\";\n    SEM_TYPES[\"ACTIVITY\"] = \"activity\";\n    SEM_TYPES[\"ACTIVITY_SCALED\"] = \"activityScaled\";\n    SEM_TYPES[\"MACROMOLECULE\"] = \"Macromolecule\";\n})(SEM_TYPES || (SEM_TYPES = {}));\nexport const MSA = 'MSA';\nexport const STATS = 'stats';\nexport const EMBEDDING_STATUS = 'embeddingStatus';\nexport const PEPTIDES_ANALYSIS = 'isPeptidesAnalysis';\nexport var FLAGS;\n(function (FLAGS) {\n    FLAGS[\"CELL_CHANGING\"] = \"isCellChanging\";\n})(FLAGS || (FLAGS = {}));\nexport const aarGroups = {\n    'R': 'PC', 'H': 'PC', 'K': 'PC',\n    'D': 'NC', 'E': 'NC',\n    'S': 'U', 'T': 'U', 'N': 'U', 'Q': 'U',\n    'C': 'SC', 'U': 'SC', 'G': 'SC', 'P': 'SC',\n    'A': 'H', 'V': 'H', 'I': 'H', 'L': 'H', 'M': 'H', 'F': 'H', 'Y': 'H', 'W': 'H',\n    '-': '-',\n};\nexport const groupDescription = {\n    'PC': { 'description': 'Positive Amino Acids, with Electrically Charged Side Chains', 'aminoAcids': ['R', 'H', 'K'] },\n    'NC': { 'description': 'Negative Amino Acids, with Electrically Charged Side Chains', 'aminoAcids': ['D', 'E'] },\n    'U': { 'description': 'Amino Acids with Polar Uncharged Side Chains', 'aminoAcids': ['S', 'T', 'N', 'Q'] },\n    'SC': { 'description': 'Special Cases', 'aminoAcids': ['C', 'U', 'G', 'P'] },\n    'H': {\n        'description': 'Amino Acids with Hydrophobic Side Chain',\n        'aminoAcids': ['A', 'V', 'I', 'L', 'M', 'F', 'Y', 'W'],\n    },\n    '-': { 'description': 'Unknown Amino Acid', 'aminoAcids': ['-'] },\n};\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as ui from 'datagrok-api/ui';\nimport * as grok from 'datagrok-api/grok';\nimport $ from 'cash-dom';\nimport { NotationConverter } from '@datagrok-libraries/bio/src/utils/notation-converter';\nlet convertDialog = null;\nlet convertDialogSubs = [];\n/**\n * Converts notations of a Macromolecule column\n *\n * @param {DG.column} col Column with 'Macromolecule' semantic type\n */\nexport function convert(col) {\n    const converter = new NotationConverter(col);\n    const currentNotation = converter.notation;\n    //TODO: read all notations\n    const notations = [\n        \"fasta\" /* NOTATION.FASTA */,\n        \"separator\" /* NOTATION.SEPARATOR */,\n        \"helm\" /* NOTATION.HELM */\n    ];\n    const separatorArray = ['-', '.', '/'];\n    const filteredNotations = notations.filter((e) => e !== currentNotation);\n    const targetNotationInput = ui.choiceInput('Convert to', filteredNotations[0], filteredNotations);\n    const separatorInput = ui.choiceInput('Separator', separatorArray[0], separatorArray);\n    // hide the separator input for non-SEPARATOR target notations\n    const toggleSeparator = () => {\n        if (targetNotationInput.value !== \"separator\" /* NOTATION.SEPARATOR */)\n            $(separatorInput.root).hide();\n        else\n            $(separatorInput.root).show();\n    };\n    // set correct visibility on init\n    toggleSeparator();\n    targetNotationInput.onChanged(() => {\n        toggleSeparator();\n    });\n    if (convertDialog == null) {\n        convertDialog = ui.dialog('Convert sequence notation')\n            .add(ui.div([\n            ui.divText('Current notation: ' + currentNotation, {\n                style: {\n                    'text-align': 'center',\n                    'font-weight': 'bold',\n                    'font-size': '14px',\n                    'padding': '5px',\n                }\n            }),\n            targetNotationInput.root,\n            separatorInput.root\n        ]))\n            .onOK(() => __awaiter(this, void 0, void 0, function* () {\n            const targetNotation = targetNotationInput.value;\n            const separator = separatorInput.value;\n            yield convertDo(col, targetNotation, separator);\n        }))\n            .show({ x: 350, y: 100 });\n        convertDialogSubs.push(convertDialog.onClose.subscribe((value) => {\n            convertDialogSubs.forEach((s) => { s.unsubscribe(); });\n            convertDialogSubs = [];\n            convertDialog = null;\n        }));\n    }\n}\nexport function convertDo(srcCol, targetNotation, separator) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const converter = new NotationConverter(srcCol);\n        const newColumn = converter.convert(targetNotation, separator);\n        srcCol.dataFrame.columns.add(newColumn);\n        // call to calculate 'cell.renderer' tag\n        yield grok.data.detectSemanticTypes(srcCol.dataFrame);\n        return newColumn;\n    });\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport { FastaFileHandler } from '@datagrok-libraries/bio/src/utils/fasta-handler';\n//@ts-ignore\nimport Aioli from '@biowasm/aioli';\nimport { AlignedSequenceEncoder } from '@datagrok-libraries/bio/src/sequence-encoder';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\n/**\n * Converts array of sequences into simple fasta string.\n *\n * @param {string[]} sequences Input list of sequences.\n * @return {string} Fasta-formatted string.\n */\nfunction _stringsToFasta(sequences) {\n    return sequences.reduce((a, v, i) => a + `>sample${i + 1}\\n${v}\\n`, '');\n}\n/**\n * Runs Aioli environment with kalign tool.\n *\n * @param {DG.Column} srcCol Column with sequences.\n * @param {boolean} isAligned Whether the column is aligned.\n * @param {string} unUsedName\n * @return {Promise<DG.Column>} Aligned sequences.\n */\nexport function runKalign(srcCol, isAligned = false, unUsedName = '') {\n    return __awaiter(this, void 0, void 0, function* () {\n        let sequences = srcCol.toList();\n        if (isAligned)\n            sequences = sequences.map((v, _) => AlignedSequenceEncoder.clean(v).replace(/\\-/g, ''));\n        const fasta = _stringsToFasta(sequences);\n        const CLI = yield new Aioli({\n            tool: 'kalign',\n            version: '3.3.1',\n            reinit: true,\n        });\n        console.log(['fasta.length =', fasta.length]);\n        yield CLI.fs.writeFile('input.fa', fasta);\n        const output = yield CLI.exec('kalign input.fa -f fasta -o result.fasta');\n        console.warn(output);\n        const buf = yield CLI.cat('result.fasta');\n        if (!buf)\n            throw new Error(`kalign output no result`);\n        const ffh = new FastaFileHandler(buf);\n        const aligned = ffh.sequencesArray; // array of sequences extracted from FASTA\n        const tgtCol = DG.Column.fromStrings(unUsedName, aligned);\n        // units\n        const srcUnits = srcCol.getTag(DG.TAGS.UNITS);\n        //aligned\n        const srcAligned = srcCol.getTag(UnitsHandler.TAGS.aligned);\n        const tgtAligned = srcAligned + '.MSA';\n        //alphabet\n        const srcAlphabet = srcCol.getTag(UnitsHandler.TAGS.alphabet);\n        tgtCol.setTag(DG.TAGS.UNITS, srcUnits);\n        tgtCol.setTag(UnitsHandler.TAGS.aligned, tgtAligned);\n        tgtCol.setTag(UnitsHandler.TAGS.alphabet, srcAlphabet);\n        tgtCol.semType = DG.SEMTYPE.MACROMOLECULE;\n        return tgtCol;\n    });\n}\nexport function testMSAEnoughMemory(col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const sequencesCount = col.length;\n        const delta = sequencesCount / 100;\n        for (let i = delta; i < sequencesCount; i += delta) {\n            try {\n                yield runKalign(DG.Column.fromStrings(col.name, col.toList().slice(0, Math.round(i))));\n                console.log(`runKalign succeeded on ${i}`);\n            }\n            catch (error) {\n                console.log(`runKalign failed on ${i} with '${error}'`);\n            }\n        }\n    });\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\nimport { getSimilarityFromDistance } from '@datagrok-libraries/utils/src/similarity-metrics';\nimport { AvailableMetrics } from '@datagrok-libraries/ml/src/typed-metrics';\nimport * as grok from 'datagrok-api/grok';\nexport function getDistances(col, seq) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const stringArray = col.toList();\n        const distances = new Array(stringArray.length).fill(0);\n        for (let i = 0; i < stringArray.length; ++i) {\n            const distance = stringArray[i] ? AvailableMetrics['String']['Levenshtein'](stringArray[i], seq) : null;\n            distances[i] = distance ? distance / Math.max(stringArray[i].length, seq.length) : null;\n        }\n        return distances;\n    });\n}\nexport function getSimilaritiesMarix(dim, seqCol, df, colName, simArr) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const distances = new Array(simArr.length).fill(null);\n        for (let i = 0; i != dim - 1; ++i) {\n            const seq = seqCol.get(i);\n            df.rows.removeAt(0, 1, false);\n            distances[i] = (yield getDistances(df.col(colName), seq));\n        }\n        for (let i = 0; i < distances.length; i++) {\n            for (let j = 0; j < distances[i].length; j++) {\n                distances[i][j] = getSimilarityFromDistance(distances[i][j]);\n            }\n            simArr[i] = DG.Column.fromList(DG.COLUMN_TYPE.FLOAT, 'distances', distances[i]);\n        }\n        return simArr;\n    });\n}\nexport function createTooltipElement(params) {\n    const tooltipElement = ui.divH([]);\n    const columnNames = ui.divV([\n        ui.divText(params.seqCol.name),\n        ui.divText(params.activityCol.name),\n    ]);\n    columnNames.style.fontWeight = 'bold';\n    columnNames.style.display = 'flex';\n    columnNames.style.justifyContent = 'space-between';\n    tooltipElement.append(columnNames);\n    params.line.mols.forEach((molIdx, idx) => {\n        const activity = ui.divText(params.activityCol.get(molIdx).toFixed(2));\n        activity.style.display = 'flex';\n        activity.style.justifyContent = 'left';\n        activity.style.paddingLeft = '30px';\n        tooltipElement.append(ui.divV([\n            ui.divText(params.seqCol.get(molIdx)),\n            activity,\n        ]));\n    });\n    return tooltipElement;\n}\nfunction moleculeInfo(df, idx, seqColName) {\n    let dict = {};\n    for (let col of df.columns) {\n        if (col.name !== seqColName) {\n            dict[col.name] = df.get(col.name, idx);\n        }\n    }\n    return ui.tableFromMap(dict);\n}\nexport function createPropPanelElement(params) {\n    const propPanel = ui.divV([]);\n    const columnNames = ui.divH([\n        ui.divText(params.seqCol.name),\n        ui.divText(params.activityCol.name),\n    ]);\n    columnNames.style.fontWeight = 'bold';\n    columnNames.style.justifyContent = 'space-between';\n    propPanel.append(columnNames);\n    const hosts = [];\n    params.line.mols.forEach((molIdx, hostIdx) => {\n        const activity = ui.divText(params.activityCol.get(molIdx).toFixed(2));\n        activity.style.paddingLeft = '15px';\n        activity.style.paddingLeft = '10px';\n        const molHost = ui.divText(params.seqCol.get(molIdx));\n        if (params.df.currentRowIdx === molIdx) {\n            molHost.style.border = 'solid 1px lightgrey';\n        }\n        //@ts-ignore\n        ui.tooltip.bind(molHost, () => moleculeInfo(params.df, molIdx, params.seqCol.name));\n        molHost.onclick = () => {\n            const obj = grok.shell.o;\n            molHost.style.border = 'solid 1px lightgrey';\n            params.df.currentRowIdx = molIdx;\n            hosts.forEach((h, i) => {\n                if (i !== hostIdx) {\n                    h.style.border = '';\n                }\n            });\n            setTimeout(() => {\n                grok.shell.o = obj;\n            }, 1000);\n        };\n        propPanel.append(ui.divH([\n            molHost,\n            activity,\n        ]));\n        hosts.push(molHost);\n    });\n    propPanel.append(ui.divH([\n        ui.divText(`Cliff: `, { style: { fontWeight: 'bold', paddingRight: '5px' } }),\n        ui.divText(params.sali.toFixed(2))\n    ]));\n    return propPanel;\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as DG from 'datagrok-api/dg';\nimport { reduceDimensinalityWithNormalization } from '@datagrok-libraries/ml/src/sequence-space';\nexport function sequenceSpace(spaceParams) {\n    return __awaiter(this, void 0, void 0, function* () {\n        // code deprecated since seqCol is encoded\n        /*    let preparedData: any;\n          if (!(spaceParams.seqCol!.tags[DG.TAGS.UNITS] === 'HELM')) {\n            const sep = spaceParams.seqCol.getTag(UnitsHandler.TAGS.separator);\n            const sepFinal = sep ? sep === '.' ? '\\\\\\.' : sep : '-';\n            const regex = new RegExp(sepFinal, 'g');\n            if (Object.keys(AvailableMetrics['String']).includes(spaceParams.similarityMetric))\n              preparedData = spaceParams.seqCol.toList().map((v: string) => v.replace(regex, '')) as string[];\n            else\n              preparedData = spaceParams.seqCol.toList().map((v: string) => v.replace(regex, '')) as string[];\n          } else {\n            preparedData = spaceParams.seqCol.toList();\n          }  */\n        const sequenceSpaceResult = yield reduceDimensinalityWithNormalization(spaceParams.seqCol.toList(), spaceParams.methodName, spaceParams.similarityMetric, spaceParams.options);\n        const cols = spaceParams.embedAxesNames.map((name, index) => DG.Column.fromFloat32Array(name, sequenceSpaceResult.embedding[index]));\n        return { distance: sequenceSpaceResult.distance, coordinates: new DG.ColumnList(cols) };\n    });\n}\nexport function getEmbeddingColsNames(df) {\n    const axes = ['Embed_X', 'Embed_Y'];\n    const colNameInd = df.columns.names().filter((it) => it.includes(axes[0])).length + 1;\n    return axes.map((it) => `${it}_${colNameInd}`);\n}\n","import * as DG from 'datagrok-api/dg';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport * as grok from 'datagrok-api/grok';\nimport { CAP_GROUP_NAME, CAP_GROUP_SMILES, jsonSdfMonomerLibDict, MONOMER_ENCODE_MAX, MONOMER_ENCODE_MIN, MONOMER_SYMBOL, RGROUP_ALTER_ID, RGROUP_FIELD, RGROUP_LABEL, SDF_MONOMER_NAME } from '../const';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nexport const HELM_CORE_LIB_FILENAME = '/samples/HELMCoreLibrary.json';\nexport const HELM_CORE_LIB_MONOMER_SYMBOL = 'symbol';\nexport const HELM_CORE_LIB_MOLFILE = 'molfile';\nexport const HELM_CORE_FIELDS = ['symbol', 'molfile', 'rgroups', 'name'];\nexport function encodeMonomers(col) {\n    let encodeSymbol = MONOMER_ENCODE_MIN;\n    const monomerSymbolDict = {};\n    const units = col.tags[DG.TAGS.UNITS];\n    const sep = col.getTag(UnitsHandler.TAGS.separator);\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const encodedStringArray = [];\n    for (let i = 0; i < col.length; ++i) {\n        let encodedMonomerStr = '';\n        const monomers = splitterFunc(col.get(i));\n        monomers.forEach((m) => {\n            if (!monomerSymbolDict[m]) {\n                if (encodeSymbol > MONOMER_ENCODE_MAX) {\n                    grok.shell.error(`Not enough symbols to encode monomers`);\n                    return null;\n                }\n                monomerSymbolDict[m] = encodeSymbol;\n                encodeSymbol++;\n            }\n            encodedMonomerStr += String.fromCodePoint(monomerSymbolDict[m]);\n        });\n        encodedStringArray.push(encodedMonomerStr);\n    }\n    return DG.Column.fromStrings('encodedMolecules', encodedStringArray);\n}\nexport function getMolfilesFromSeq(col, monomersLibObject) {\n    const units = col.tags[DG.TAGS.UNITS];\n    const sep = col.getTag('separator');\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const monomersDict = createMomomersMolDict(monomersLibObject);\n    const molFiles = [];\n    for (let i = 0; i < col.length; ++i) {\n        const macroMolecule = col.get(i);\n        const monomers = splitterFunc(macroMolecule);\n        const molFilesForSeq = [];\n        for (let j = 0; j < monomers.length; ++j) {\n            if (monomers[j]) {\n                if (!monomersDict[monomers[j]]) {\n                    grok.shell.warning(`Monomer ${monomers[j]} is missing in HELM library. Structure cannot be created`);\n                    return null;\n                }\n                molFilesForSeq.push(JSON.parse(JSON.stringify(monomersDict[monomers[j]])));\n            }\n        }\n        molFiles.push(molFilesForSeq);\n    }\n    return molFiles;\n}\nexport function getMolfilesFromSingleSeq(cell, monomersLibObject) {\n    const units = cell.column.tags[DG.TAGS.UNITS];\n    const sep = cell.column.getTag('separator');\n    const splitterFunc = WebLogo.getSplitter(units, sep);\n    const monomersDict = createMomomersMolDict(monomersLibObject);\n    const molFiles = [];\n    const macroMolecule = cell.value;\n    const monomers = splitterFunc(macroMolecule);\n    const molFilesForSeq = [];\n    for (let j = 0; j < monomers.length; ++j) {\n        if (monomers[j]) {\n            if (!monomersDict[monomers[j]]) {\n                grok.shell.warning(`Monomer ${monomers[j]} is missing in HELM library. Structure cannot be created`);\n                return null;\n            }\n            molFilesForSeq.push(JSON.parse(JSON.stringify(monomersDict[monomers[j]])));\n        }\n    }\n    molFiles.push(molFilesForSeq);\n    return molFiles;\n}\nexport function createMomomersMolDict(lib) {\n    const dict = {};\n    lib.forEach((it) => {\n        if (it['polymerType'] === 'PEPTIDE') {\n            const monomerObject = {};\n            HELM_CORE_FIELDS.forEach((field) => {\n                monomerObject[field] = it[field];\n            });\n            dict[it[HELM_CORE_LIB_MONOMER_SYMBOL]] = monomerObject;\n        }\n    });\n    return dict;\n}\nexport function createJsonMonomerLibFromSdf(table) {\n    const resultLib = [];\n    for (let i = 0; i < table.rowCount; i++) {\n        const monomer = {};\n        Object.keys(jsonSdfMonomerLibDict).forEach((key) => {\n            if (key === MONOMER_SYMBOL) {\n                const monomerSymbol = table.get(jsonSdfMonomerLibDict[key], i);\n                monomer[key] = monomerSymbol === '.' ? table.get(SDF_MONOMER_NAME, i) : monomerSymbol;\n            }\n            else if (key === RGROUP_FIELD) {\n                const rgroups = table.get(jsonSdfMonomerLibDict[key], i).split('\\n');\n                const jsonRgroups = [];\n                rgroups.forEach((g) => {\n                    const rgroup = {};\n                    const altAtom = g.substring(g.lastIndexOf(']') + 1);\n                    const radicalNum = g.match(/\\[R(\\d+)\\]/)[1];\n                    rgroup[CAP_GROUP_SMILES] = altAtom === 'H' ? `[*:${radicalNum}][H]` : `O[*:${radicalNum}]`;\n                    rgroup[RGROUP_ALTER_ID] = altAtom === 'H' ? `R${radicalNum}-H` : `R${radicalNum}-OH`;\n                    rgroup[CAP_GROUP_NAME] = altAtom === 'H' ? `H` : `OH`;\n                    rgroup[RGROUP_LABEL] = `R${radicalNum}`;\n                    jsonRgroups.push(rgroup);\n                });\n                monomer[key] = jsonRgroups;\n            }\n            else {\n                if (jsonSdfMonomerLibDict[key])\n                    monomer[key] = table.get(jsonSdfMonomerLibDict[key], i);\n            }\n        });\n        resultLib.push(monomer);\n    }\n    return resultLib;\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport { VdRegionType } from '@datagrok-libraries/bio/src/vd-regions';\nconst vrt = VdRegionType;\n// Positions of regions for numbering schemes\n// http://www.bioinf.org.uk/abs/info.html\n// const imgtRegions: VdRegion[] = [\n//   new VdRegion(vrt.FR, 'FR1', 'Light', 1, '1', '26'),\n//   new VdRegion(vrt.FR, 'FR1', 'Heavy', 1, '1', '26'),\n//\n//   new VdRegion(vrt.CDR, 'CDR1', 'Light', 2, '27', '38'), // 27-32\n//   new VdRegion(vrt.CDR, 'CDR1', 'Heavy', 2, '27', '38'), // 27-32\n//\n//   new VdRegion(vrt.FR, 'FR2', 'Light', 3, '39', '55'),\n//   new VdRegion(vrt.FR, 'FR2', 'Heavy', 3, '39', '55'),\n//\n//   new VdRegion(vrt.CDR, 'CDR2', 'Light', 4, '56', '65'),\n//   new VdRegion(vrt.CDR, 'CDR2', 'Heavy', 4, '56', '65'),\n//\n//   new VdRegion(vrt.FR, 'FR3', 'Light', 5, '66', '104'),\n//   new VdRegion(vrt.FR, 'FR3', 'Heavy', 5, '66', '104'),\n//\n//   new VdRegion(vrt.CDR, 'CDR3', 'Light', 6, '105', '117'),\n//   new VdRegion(vrt.CDR, 'CDR3', 'Heavy', 6, '105', '117'),\n//\n//   new VdRegion(vrt.FR, 'FR4', 'Light', 7, '118', null/*127*/),\n//   new VdRegion(vrt.FR, 'FR4', 'Heavy', 7, '118', null/*128*/),\n// ];\n/** Viewer with tabs based on description of chain regions.\n *  Used to define regions of an immunoglobulin LC.\n */\nexport class VdRegionsViewer extends DG.JsViewer {\n    constructor() {\n        super();\n        // private regionsDf: DG.DataFrame;\n        this.regionsFg = null;\n        // private regionsTV: DG.TableView;\n        this.regionsRoot = null;\n        this.isOpened = false;\n        this.panelNode = null;\n        this.regions = [];\n        // #region -- Handle controls' events --\n        this.resizing = false;\n        // #endregion\n        //#region -- View --\n        this.host = null;\n        this.mainLayout = null;\n        this.logos = [];\n        // To prevent ambiguous numbering scheme in MLB\n        this.regionTypes = this.stringList('regionTypes', [vrt.CDR], { choices: Object.values(vrt).filter((t) => t != vrt.Unknown) });\n        this.chains = this.stringList('chains', ['Heavy', 'Light'], { choices: ['Heavy', 'Light'] });\n        this.sequenceColumnNamePostfix = this.string('sequenceColumnNamePostfix', 'chain sequence');\n        this.skipEmptyPositions = this.bool('skipEmptyPositions', false);\n        this.positionWidth = this.float('positionWidth', 16);\n    }\n    get df() {\n        return this.dataFrame;\n    }\n    // TODO: .onTableAttached is not calling on dataFrame set, onPropertyChanged  also not calling\n    setDf(value, regions) {\n        return __awaiter(this, void 0, void 0, function* () {\n            console.debug('VdRegionsViewer.setDf()');\n            yield this.destroyView();\n            this.regions = regions;\n            this.dataFrame = value;\n            yield this.buildView();\n        });\n    }\n    init() {\n        return __awaiter(this, void 0, void 0, function* () {\n            //#region regionsDF with filter\n            // this.regionsDf = DG.DataFrame.fromObjects(this.regions);\n            // this.regionsDf.rows.filter((row) => row.name == 'CDR1');\n            // // To available options /\n            // this.regionsFg = (await this.regionsDf.plot.fromType(DG.VIEWER.FILTERS, {\n            //   // columnNames: ['name',],\n            //   showFilterCountsIndication: false,\n            //   showHeader: false,\n            //   showSearchBox: false,\n            //   filters: [\n            //     {type: DG.FILTER_TYPE.CATEGORICAL, column: 'type', label: 'Region name', showHistogram: false},\n            //     {type: DG.FILTER_TYPE.CATEGORICAL, column: 'name', label: 'Region type', showHistogram: false},\n            //   ],\n            //   title: 'Regions filter',\n            //   showTitle: true,\n            //   description: 'Filter for regions of multiple alignment by IMGT nomenclature',\n            //   someProperty: 'Hello',\n            // })) as DG.FilterGroup;\n            //#endregion regionsDF with filter\n            // this.mlbView.dockManager.dock(this.regionsFg.root, DG.DOCK_TYPE.LEFT, rootNode, 'Filter regions', 0.2);\n            this.subs.push(ui.onSizeChanged(this.root).subscribe(this.rootOnSizeChanged.bind(this)));\n            // rxjs.fromEvent(this.root, 'mousemove').subscribe(this.onMouseMoveRoot.bind(this));\n            this.root.addEventListener('mousemove', this.onMouseMoveRoot.bind(this));\n            yield this.buildView();\n        });\n    }\n    onTableAttached() {\n        return __awaiter(this, void 0, void 0, function* () {\n            yield this.init();\n        });\n    }\n    onPropertyChanged(property) {\n        const _super = Object.create(null, {\n            onPropertyChanged: { get: () => super.onPropertyChanged }\n        });\n        return __awaiter(this, void 0, void 0, function* () {\n            _super.onPropertyChanged.call(this, property);\n            if (property) {\n                switch (property.name) {\n                    case 'regionTypes':\n                        break;\n                    case 'chains':\n                        break;\n                    case 'sequenceColumnNamePostfix':\n                        break;\n                    case 'skipEmptyPositions':\n                        // for (let orderI = 0; orderI < this.logos.length; orderI++) {\n                        //   for (let chainI = 0; chainI < this.chains.length; chainI++) {\n                        //     const chain: string = this.chains[chainI];\n                        //     this.logos[orderI][chain].setOptions({skipEmptyPositions: this.skipEmptyPositions});\n                        //   }\n                        // }\n                        // this.calcSize();\n                        yield this.destroyView();\n                        yield this.buildView();\n                        break;\n                    case 'positionWidth':\n                        yield this.destroyView();\n                        yield this.buildView();\n                        break;\n                }\n            }\n        });\n    }\n    reset() {\n        return __awaiter(this, void 0, void 0, function* () {\n        });\n    }\n    open(mlbView) {\n        return __awaiter(this, void 0, void 0, function* () {\n            if (!this.isOpened) {\n                this.isOpened = true;\n                this.panelNode = mlbView.dockManager.dock(this.root, DG.DOCK_TYPE.TOP, null, 'Regions', 0.2);\n            }\n        });\n    }\n    show(mlbView) {\n        return __awaiter(this, void 0, void 0, function* () {\n        });\n    }\n    rootOnSizeChanged(args) {\n        this.calcSize();\n    }\n    destroyView() {\n        return __awaiter(this, void 0, void 0, function* () {\n            // TODO: Unsubscribe from and remove all view elements\n            console.debug(`VdRegionsViewer.destroyView( mainLayout = ${!this.mainLayout ? 'none' : 'value'} )`);\n            if (this.mainLayout != null) {\n                // this.root.removeChild(this.host);\n                this.mainLayout.remove();\n                this.host.remove();\n                this.host = null;\n                this.mainLayout = null;\n            }\n        });\n    }\n    buildView() {\n        return __awaiter(this, void 0, void 0, function* () {\n            console.debug('VdRegionsViewer.buildView() start');\n            const colNames = Object.assign({}, ...this.chains.map((chain) => ({ [chain]: `${chain} ${this.sequenceColumnNamePostfix}` })));\n            const regionsFiltered = this.regions.filter((r) => this.regionTypes.includes(r.type));\n            const orderList = Array.from(new Set(regionsFiltered.map((r) => r.order))).sort();\n            this.logos = [];\n            for (let orderI = 0; orderI < orderList.length; orderI++) {\n                const regionChains = {};\n                for (const chain of this.chains) {\n                    const region = regionsFiltered\n                        .find((r) => r.order == orderList[orderI] && r.chain == chain);\n                    regionChains[chain] = (yield this.dataFrame.plot.fromType('WebLogo', {\n                        sequenceColumnName: colNames[chain],\n                        startPositionName: region.positionStartName,\n                        endPositionName: region.positionEndName,\n                        fixWidth: true,\n                        skipEmptyPositions: this.skipEmptyPositions,\n                        positionWidth: this.positionWidth,\n                    }));\n                }\n                // WebLogo creation fires onRootSizeChanged event even before control being added to this.logos\n                this.logos[orderI] = regionChains;\n            }\n            // ui.tableFromMap()\n            // DG.HtmlTable.create()\n            this.mainLayout = ui.table(this.chains, (chain) => {\n                const elements = [\n                    // This is chain label\n                    ...(orderList.length > 0 ? [ui.div(chain, {\n                            style: {\n                                transform: 'rotate(-90deg)',\n                                font: '12px Roboto, Roboto Local, sans-serif',\n                                textAlign: 'center',\n                                width: '16px',\n                                marginTop: '24px',\n                                marginLeft: '6px',\n                            }\n                        })] : []),\n                    // List with controls for regions\n                    ...[...Array(orderList.length).keys()].map((orderI) => {\n                        const wl = this.logos[orderI][chain];\n                        wl.root.style.height = '100%';\n                        const resDiv = ui.div([wl.root] /*`${chain} ${regionsFiltered[rI]}`*/, {\n                            style: {\n                                // height: '100%',\n                                marginTop: '4px',\n                                marginBottom: '4px',\n                            }\n                        });\n                        return resDiv;\n                    })\n                ];\n                return elements;\n            }, ['', ...[...Array(orderList.length).keys()].map((orderI) => regionsFiltered.find((r) => r.order == orderList[orderI] && r.chain == this.chains[0]).name || 'Name')]);\n            this.mainLayout.className = 'mlb-vd-regions-viewer-table2';\n            // this.mainLayout.style.background = '#EEEEFF';\n            // this.mainLayout.style.height = '100%';\n            // this.mainLayout.style.border = '1px solid black';\n            const color = `#ffbb${Math.ceil(Math.random() * 255).toString(16)}`;\n            this.host = ui.box(this.mainLayout, { /*style: {backgroundColor: color}*/});\n            this.root.appendChild(this.host);\n            this.root.style.overflowX = 'auto';\n            this.calcSize();\n            console.debug('VdRegionsViewer.buildView() end');\n        });\n    }\n    calcSize() {\n        const logoHeight = (this.root.clientHeight - 54) / this.chains.length;\n        const maxHeight = Math.min(logoHeight, Math.max(...this.logos.map((wlDict) => Math.max(...Object.values(wlDict).map((wl) => wl.maxHeight)))));\n        for (let orderI = 0; orderI < this.logos.length; orderI++) {\n            for (let chainI = 0; chainI < this.chains.length; chainI++) {\n                const chain = this.chains[chainI];\n                this.logos[orderI][chain].root.style.height = `${maxHeight}px`;\n            }\n        }\n    }\n    //#endregion -- View --\n    onMouseMoveRoot(e) {\n        // ui.tooltip.show('text', e.x + 8, e.y + 8,);\n        // console.log(`onMouseMoveRoot.( x: ${e.x}, y: ${e.y} )`);\n    }\n}\n","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nimport { getMolfilesFromSingleSeq } from '../utils/utils';\nimport { getMacroMol } from '../utils/atomic-works';\n/**\n * 3D representation widget of macromolecule.\n *\n * @export\n * @param {DG.Cell} macroMolecule macromolecule cell.\n * @param {any[]} monomersLibObject\n * @return {Promise<DG.Widget>} Widget.\n */\nexport function representationsWidget(macroMolecule, monomersLibObject) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const pi = DG.TaskBarProgressIndicator.create('Creating 3D view');\n        let widgetHost;\n        let molBlock3D = '';\n        try {\n            try {\n                const atomicCodes = getMolfilesFromSingleSeq(macroMolecule, monomersLibObject);\n                const result = yield getMacroMol(atomicCodes);\n                const molBlock2D = result[0];\n                molBlock3D = (yield grok.functions.call('Bio:Embed', { molBlock2D }));\n            }\n            catch (e) {\n                console.warn(e);\n            }\n            try {\n                molBlock3D = molBlock3D.replaceAll('\\\\n', '\\n');\n                const stringBlob = new Blob([molBlock3D], { type: 'text/plain' });\n                const nglHost = ui.div([], { classes: 'd4-ngl-viewer', id: 'ngl-3d-host' });\n                //@ts-ignore\n                const stage = new NGL.Stage(nglHost, { backgroundColor: 'white' });\n                //@ts-ignore\n                stage.loadFile(stringBlob, { ext: 'sdf' }).then(function (comp) {\n                    stage.setSize(300, 300);\n                    comp.addRepresentation('ball+stick');\n                    comp.autoView();\n                });\n                const sketch = grok.chem.svgMol(molBlock3D);\n                const panel = ui.divH([sketch]);\n                widgetHost = ui.div([panel, nglHost]);\n            }\n            catch (e) {\n                widgetHost = ui.divText('Couldn\\'t get peptide structure');\n            }\n        }\n        catch (e) {\n            widgetHost = ui.divText('Couldn\\'t get peptide structure');\n        }\n        pi.close();\n        return new DG.Widget(widgetHost);\n    });\n}\n","module.exports = $;","module.exports = DG;","module.exports = grok;","module.exports = rxjs;","module.exports = ui;","module.exports = wu;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","// This function allow to reference async chunks\n__webpack_require__.u = (chunkId) => {\n\t// return url for filenames based on template\n\treturn \"\" + chunkId + \".js\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","var scriptUrl;\nif (__webpack_require__.g.importScripts) scriptUrl = __webpack_require__.g.location + \"\";\nvar document = __webpack_require__.g.document;\nif (!scriptUrl && document) {\n\tif (document.currentScript)\n\t\tscriptUrl = document.currentScript.src\n\tif (!scriptUrl) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tif(scripts.length) scriptUrl = scripts[scripts.length - 1].src\n\t}\n}\n// When supporting browsers where an automatic publicPath is not supported you must specify an output.publicPath manually via configuration\n// or pass an empty string (\"\") and set the __webpack_public_path__ variable from your code to use your own logic.\nif (!scriptUrl) throw new Error(\"Automatic publicPath is not supported in this browser\");\nscriptUrl = scriptUrl.replace(/#.*$/, \"\").replace(/\\?.*$/, \"\").replace(/\\/[^\\/]+$/, \"/\");\n__webpack_require__.p = scriptUrl;","__webpack_require__.b = document.baseURI || self.location.href;\n\n// object to store loaded and loading chunks\n// undefined = chunk not loaded, null = chunk preloaded/prefetched\n// [resolve, reject, Promise] = chunk loading, 0 = chunk loaded\nvar installedChunks = {\n\t\"package\": 0\n};\n\n// no chunk on demand loading\n\n// no prefetching\n\n// no preloaded\n\n// no HMR\n\n// no HMR manifest\n\n// no on chunks loaded\n\n// no jsonp function","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\n/* Do not change these import lines to match external modules in webpack configuration */\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\nimport * as DG from 'datagrok-api/dg';\nexport const _package = new DG.Package();\nimport { MacromoleculeDifferenceCellRenderer, MonomerCellRenderer } from './utils/cell-renderer';\nimport { WebLogo } from '@datagrok-libraries/bio/src/viewers/web-logo';\nimport { VdRegionsViewer } from './viewers/vd-regions-viewer';\nimport { runKalign } from './utils/multiple-sequence-alignment';\nimport { SequenceAlignment } from './seq_align';\nimport { getEmbeddingColsNames, sequenceSpace } from './utils/sequence-space';\nimport { getActivityCliffs } from '@datagrok-libraries/ml/src/viewers/activity-cliffs';\nimport { createPropPanelElement, createTooltipElement, getSimilaritiesMarix } from './utils/sequence-activity-cliffs';\nimport { createJsonMonomerLibFromSdf, encodeMonomers, getMolfilesFromSeq, HELM_CORE_LIB_FILENAME } from './utils/utils';\nimport { getMacroMol } from './utils/atomic-works';\nimport { MacromoleculeSequenceCellRenderer } from './utils/cell-renderer';\nimport { convert } from './utils/convert';\nimport { representationsWidget } from './widgets/representations';\nimport { UnitsHandler } from '@datagrok-libraries/bio/src/utils/units-handler';\nimport { FastaFileHandler } from '@datagrok-libraries/bio/src/utils/fasta-handler';\nimport { removeEmptyStringRows } from '@datagrok-libraries/utils/src/dataframe-utils';\nimport { splitAlignedSequences } from '@datagrok-libraries/bio/src/utils/splitter';\nimport * as C from './utils/constants';\nimport { getFingerprints } from './calculations/fingerprints';\n//tags: init\nexport function initBio() {\n    return __awaiter(this, void 0, void 0, function* () {\n    });\n}\n//name: fastaSequenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: Sequence\n//meta.columnTags: units=fasta\n//output: grid_cell_renderer result\nexport function fastaSequenceCellRenderer() {\n    return new MacromoleculeSequenceCellRenderer();\n}\n//name: separatorSequenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: Sequence\n//meta.columnTags: units=separator\n//output: grid_cell_renderer result\nexport function separatorSequenceCellRenderer() {\n    return new MacromoleculeSequenceCellRenderer();\n}\nfunction checkInputColumnUi(col, name, allowedNotations = [], allowedAlphabets = []) {\n    const [res, msg] = checkInputColumn(col, name, allowedNotations, allowedAlphabets);\n    if (!res)\n        grok.shell.warning(msg);\n    return res;\n}\nexport function checkInputColumn(col, name, allowedNotations = [], allowedAlphabets = []) {\n    let res = true;\n    let msg = '';\n    const uh = new UnitsHandler(col);\n    if (col.semType !== DG.SEMTYPE.MACROMOLECULE) {\n        grok.shell.warning(name + ' analysis is allowed for Macromolecules semantic type');\n        res = false;\n    }\n    else {\n        const notation = uh.notation;\n        if (allowedNotations.length > 0 &&\n            !allowedNotations.some((n) => notation.toUpperCase() == (n.toUpperCase()))) {\n            const notationAdd = allowedNotations.length == 0 ? 'any notation' :\n                (`notation${allowedNotations.length > 1 ? 's' : ''} ${allowedNotations.map((n) => `\"${n}\"`).join(', ')} `);\n            msg = `${name} + ' analysis is allowed for Macromolecules with notation ${notationAdd}.`;\n            res = false;\n        }\n        else if (!uh.isHelm()) {\n            // alphabet is not specified for 'helm' notation\n            const alphabet = uh.alphabet;\n            if (allowedAlphabets.length > 0 &&\n                !allowedAlphabets.some((a) => alphabet.toUpperCase() == (a.toUpperCase()))) {\n                const alphabetAdd = allowedAlphabets.length == 0 ? 'any alphabet' :\n                    (`alphabet${allowedAlphabets.length > 1 ? 's' : ''} ${allowedAlphabets.map((a) => `\"${a}\"`).join(', ')}.`);\n                msg = `${name} + ' analysis is allowed for Macromolecules with alphabet ${alphabetAdd}.`;\n                res = false;\n            }\n        }\n    }\n    return [res, msg];\n}\n//name: sequenceAlignment\n//input: string alignType {choices: ['Local alignment', 'Global alignment']}\n// eslint-disable-next-line max-len\n//input: string alignTable {choices: ['AUTO', 'NUCLEOTIDES', 'BLOSUM45', 'BLOSUM50','BLOSUM62','BLOSUM80','BLOSUM90','PAM30','PAM70','PAM250','SCHNEIDER','TRANS']}\n//input: double gap\n//input: string seq1\n//input: string seq2\n//output: object res\nexport function sequenceAlignment(alignType, alignTable, gap, seq1, seq2) {\n    const toAlign = new SequenceAlignment(seq1, seq2, gap, alignTable);\n    const res = alignType == 'Local alignment' ? toAlign.smithWaterman() : toAlign.needlemanWunch();\n    return res;\n}\n//name: WebLogo\n//description: WebLogo viewer\n//tags: viewer, panel\n//output: viewer result\nexport function webLogoViewer() {\n    return new WebLogo();\n}\n//name: VdRegions\n//description: V-Domain regions viewer\n//tags: viewer, panel\n//output: viewer result\nexport function vdRegionViewer() {\n    return new VdRegionsViewer();\n}\n//top-menu: Bio | Sequence Activity Cliffs...\n//name: Sequence Activity Cliffs\n//description: detect activity cliffs\n//input: dataframe table [Input data table]\n//input: column macroMolecule {semType: Macromolecule}\n//input: column activities\n//input: double similarity = 90 [Similarity cutoff]\n//input: string methodName { choices:[\"UMAP\", \"t-SNE\", \"SPE\"] }\nexport function activityCliffs(df, macroMolecule, activities, similarity, methodName) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!checkInputColumnUi(macroMolecule, 'Activity Cliffs'))\n            return;\n        const encodedCol = encodeMonomers(macroMolecule);\n        if (!encodedCol)\n            return;\n        const axesNames = getEmbeddingColsNames(df);\n        const options = {\n            'SPE': { cycles: 2000, lambda: 1.0, dlambda: 0.0005 },\n        };\n        const tags = {\n            'units': macroMolecule.tags['units'],\n            'aligned': macroMolecule.tags['aligned'],\n            'separator': macroMolecule.tags['separator'],\n            'alphabet': macroMolecule.tags['alphabet'],\n        };\n        const sp = yield getActivityCliffs(df, macroMolecule, encodedCol, axesNames, 'Activity cliffs', activities, similarity, 'Levenshtein', methodName, DG.SEMTYPE.MACROMOLECULE, tags, sequenceSpace, getSimilaritiesMarix, createTooltipElement, createPropPanelElement, options[methodName]);\n        return sp;\n    });\n}\n//top-menu: Bio | Sequence Space...\n//name: Sequence Space\n//input: dataframe table\n//input: column macroMolecule { semType: Macromolecule }\n//input: string methodName { choices:[\"UMAP\", \"t-SNE\", \"SPE\"] }\n//input: string similarityMetric { choices:[\"Levenshtein\", \"Tanimoto\"] }\n//input: bool plotEmbeddings = true\nexport function sequenceSpaceTopMenu(table, macroMolecule, methodName, similarityMetric = 'Levenshtein', plotEmbeddings) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!checkInputColumnUi(macroMolecule, 'Activity Cliffs'))\n            return;\n        const encodedCol = encodeMonomers(macroMolecule);\n        if (!encodedCol)\n            return;\n        const embedColsNames = getEmbeddingColsNames(table);\n        const withoutEmptyValues = DG.DataFrame.fromColumns([encodedCol]).clone();\n        const emptyValsIdxs = removeEmptyStringRows(withoutEmptyValues, encodedCol);\n        const chemSpaceParams = {\n            seqCol: withoutEmptyValues.col(encodedCol.name),\n            methodName: methodName,\n            similarityMetric: similarityMetric,\n            embedAxesNames: embedColsNames\n        };\n        const sequenceSpaceRes = yield sequenceSpace(chemSpaceParams);\n        const embeddings = sequenceSpaceRes.coordinates;\n        for (const col of embeddings) {\n            const listValues = col.toList();\n            emptyValsIdxs.forEach((ind) => listValues.splice(ind, 0, null));\n            table.columns.add(DG.Column.fromList('double', col.name, listValues));\n        }\n        let sp;\n        if (plotEmbeddings) {\n            for (const v of grok.shell.views) {\n                if (v.name === table.name)\n                    sp = v.scatterPlot({ x: embedColsNames[0], y: embedColsNames[1], title: 'Sequence space' });\n            }\n        }\n        return sp;\n    });\n}\n;\n//top-menu: Bio | To Atomic Level...\n//name: To Atomic Level\n//description: returns molfiles for each monomer from HELM library\n//input: dataframe df [Input data table]\n//input: column macroMolecule {semType: Macromolecule}\nexport function toAtomicLevel(df, macroMolecule) {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (DG.Func.find({ package: 'Chem', name: 'getRdKitModule' }).length === 0) {\n            grok.shell.warning('Transformation to atomic level requires package \"Chem\" installed.');\n            return;\n        }\n        if (!checkInputColumnUi(macroMolecule, 'To Atomic Level'))\n            return;\n        const monomersLibFile = yield _package.files.readAsText(HELM_CORE_LIB_FILENAME);\n        const monomersLibObject = JSON.parse(monomersLibFile);\n        const atomicCodes = getMolfilesFromSeq(macroMolecule, monomersLibObject);\n        const result = yield getMacroMol(atomicCodes);\n        const col = DG.Column.fromStrings('regenerated', result);\n        col.semType = DG.SEMTYPE.MOLECULE;\n        col.tags[DG.TAGS.UNITS] = 'molblock';\n        df.columns.add(col, true);\n        yield grok.data.detectSemanticTypes(df);\n    });\n}\n//top-menu: Bio | MSA...\n//name: MSA\n//input: dataframe table\n//input: column sequence { semType: Macromolecule, units: ['fasta'], alphabet: ['DNA', 'RNA', 'PT'] }\n//output: column result\nexport function multipleSequenceAlignmentAny(table, col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const func = DG.Func.find({ package: 'Bio', name: 'multipleSequenceAlignmentAny' })[0];\n        if (!checkInputColumnUi(col, 'MSA', ['fasta'], ['DNA', 'RNA', 'PT']))\n            return null;\n        const unUsedName = table.columns.getUnusedName(`msa(${col.name})`);\n        const msaCol = yield runKalign(col, false, unUsedName);\n        table.columns.add(msaCol);\n        // This call is required to enable cell renderer activation\n        yield grok.data.detectSemanticTypes(table);\n        // const tv: DG.TableView = grok.shell.tv;\n        // tv.grid.invalidate();\n        return msaCol;\n    });\n}\n//name: Bio | MSA\n//tags: bio, panel\n//input: column sequence { semType: Macromolecule }\n//output: column result\nexport function panelMSA(col) {\n    return __awaiter(this, void 0, void 0, function* () {\n        return multipleSequenceAlignmentAny(col.dataFrame, col);\n    });\n}\n//name: Composition Analysis\n//top-menu: Bio | Composition Analysis\n//output: viewer result\nexport function compositionAnalysis() {\n    return __awaiter(this, void 0, void 0, function* () {\n        // Higher priority for columns with MSA data to show with WebLogo.\n        const tv = grok.shell.tv;\n        const df = tv.dataFrame;\n        //@ts-ignore\n        const colList = df.columns.toList().filter((col) => {\n            if (col.semType != DG.SEMTYPE.MACROMOLECULE)\n                return false;\n            const colUH = new UnitsHandler(col);\n            // TODO: prevent for cyclic, branched or multiple chains in Helm\n            return true;\n        });\n        const handler = (col) => __awaiter(this, void 0, void 0, function* () {\n            if (!checkInputColumnUi(col, 'Composition'))\n                return;\n            const wlViewer = tv.addViewer('WebLogo', { sequenceColumnName: col.name });\n            grok.shell.tv.dockManager.dock(wlViewer, DG.DOCK_TYPE.DOWN, null, 'Composition analysis', 0.25);\n        });\n        let col = null;\n        if (colList.length == 0) {\n            grok.shell.error('Current table does not contain sequences');\n            return;\n        }\n        else if (colList.length > 1) {\n            const colListNames = colList.map((col) => col.name);\n            const colInput = ui.choiceInput('Column', colListNames[0], colListNames);\n            ui.dialog({\n                title: 'R-Groups Analysis',\n                helpUrl: '/help/domains/bio/macromolecules.md#composition-analysis'\n            })\n                .add(ui.div([\n                colInput,\n            ]))\n                .onOK(() => __awaiter(this, void 0, void 0, function* () {\n                var _a;\n                const col = (_a = colList.find((col) => col.name == colInput.value)) !== null && _a !== void 0 ? _a : null;\n                if (col)\n                    yield handler(col);\n            }))\n                .show();\n        }\n        else {\n            col = colList[0];\n        }\n        if (!col)\n            return;\n        yield handler(col);\n    });\n}\n//top-menu: Bio | Sdf to Json lib...\n//name: sdfToJsonLib\n//input: dataframe table\nexport function sdfToJsonLib(table) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const jsonMonomerLibrary = createJsonMonomerLibFromSdf(table);\n    });\n}\n//name: Representations\n//tags: panel, widgets\n//input: cell macroMolecule {semType: Macromolecule}\n//output: widget result\nexport function peptideMolecule(macroMolecule) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const monomersLibFile = yield _package.files.readAsText(HELM_CORE_LIB_FILENAME);\n        const monomersLibObject = JSON.parse(monomersLibFile);\n        return representationsWidget(macroMolecule, monomersLibObject);\n    });\n}\n//name: importFasta\n//description: Opens FASTA file\n//tags: file-handler\n//meta.ext: fasta, fna, ffn, faa, frn, fa, fst\n//input: string fileContent\n//output: list tables\nexport function importFasta(fileContent) {\n    const ffh = new FastaFileHandler(fileContent);\n    return ffh.importFasta();\n}\n//name: Bio | Convert ...\n//friendly-name: Bio | Convert\n//tags: panel, bio\n//input: column col {semType: Macromolecule}\nexport function convertPanel(col) {\n    convert(col);\n}\n//name: monomerCellRenderer\n//tags: cellRenderer\n//meta.cellType: Monomer\n//output: grid_cell_renderer result\nexport function monomerCellRenderer() {\n    return new MonomerCellRenderer();\n}\n//name: MacromoleculeDifferenceCellRenderer\n//tags: cellRenderer\n//meta.cellType: MacromoleculeDifference\n//output: grid_cell_renderer result\nexport function macromoleculeDifferenceCellRenderer() {\n    return new MacromoleculeDifferenceCellRenderer();\n}\n//name: testDetectMacromolecule\n//input: string path {choices: ['Demo:Files/', 'System:AppData/']}\n//output: dataframe result\nexport function testDetectMacromolecule(path) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const pi = DG.TaskBarProgressIndicator.create('Test detectMacromolecule...');\n        const fileList = yield grok.dapi.files.list(path, true, '');\n        //@ts-ignore\n        const fileListToTest = fileList.filter((fi) => fi.fileName.endsWith('.csv'));\n        let readyCount = 0;\n        const res = [];\n        for (const fileInfo of fileListToTest) {\n            try {\n                const csv = yield grok.dapi.files.readAsText(path + fileInfo.fullPath);\n                const df = DG.DataFrame.fromCsv(csv);\n                for (const col of df.columns) {\n                    const semType = yield grok.functions.call('Bio:detectMacromolecule', { col: col });\n                    if (semType === DG.SEMTYPE.MACROMOLECULE) {\n                        //console.warn(`file: ${fileInfo.path}, column: ${col.name}, ` +\n                        //  `semType: ${semType}, units: ${col.getTag(DG.TAGS.UNITS)}`);\n                        // console.warn('file: \"' + fileInfo.path + '\", semType: \"' + semType + '\", ' +\n                        //   'units: \"' + col.getTag(DG.TAGS.UNITS) + '\"');\n                        res.push({\n                            file: fileInfo.path, result: 'detected', column: col.name,\n                            message: `units: ${col.getTag(DG.TAGS.UNITS)}`\n                        });\n                    }\n                }\n            }\n            catch (err) {\n                // console.error('file: ' + fileInfo.path + ', error: ' + ex.toString());\n                res.push({\n                    file: fileInfo.path, result: 'error', column: null,\n                    message: err instanceof Error ? err.message : err.toString(),\n                });\n            }\n            finally {\n                readyCount += 1;\n                pi.update(100 * readyCount / fileListToTest.length, `Test ${fileInfo.fileName}`);\n            }\n        }\n        grok.shell.info('Test Demo:Files for detectMacromolecule finished.');\n        pi.close();\n        const resDf = DG.DataFrame.fromObjects(res);\n        resDf.name = `datasets_detectMacromolecule_${path}`;\n        return resDf;\n    });\n}\n//name: Bio | Split to monomers\n//tags: panel, bio\n//input: column col {semType: Macromolecule}\nexport function splitToMonomers(col) {\n    if (!col.getTag(UnitsHandler.TAGS.aligned).includes(C.MSA))\n        return grok.shell.error('Splitting is applicable only for aligned sequences');\n    const tempDf = splitAlignedSequences(col);\n    const originalDf = col.dataFrame;\n    for (const tempCol of tempDf.columns) {\n        const newCol = originalDf.columns.add(tempCol);\n        newCol.semType = C.SEM_TYPES.MONOMER;\n        newCol.setTag(DG.TAGS.CELL_RENDERER, C.SEM_TYPES.MONOMER);\n        newCol.setTag(C.TAGS.ALPHABET, col.getTag(C.TAGS.ALPHABET));\n    }\n    grok.shell.tv.grid.invalidate();\n}\n//name: Bio: getHelmMonomers\n//input: column col {semType: Macromolecule}\nexport function getHelmMonomers(seqCol) {\n    const stats = WebLogo.getStats(seqCol, 1, WebLogo.splitterAsHelm);\n    return Object.keys(stats.freq);\n}\nexport function macromoleculesFingerprints(mcol) {\n    return __awaiter(this, void 0, void 0, function* () {\n        grok.functions.call('Chem:getRdKitModule');\n        const monomers = getHelmMonomers(mcol);\n        const mols = yield grok.functions.call('HELM:getMolFiles', { mcol: mcol });\n        return getFingerprints(mols.toList(), monomers);\n    });\n}\n"],"names":[],"sourceRoot":""}
|