@datagrok-libraries/statistics 1.7.0 → 1.9.0
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/package.json +9 -6
- package/src/fit/fit-curve.d.ts +25 -66
- package/src/fit/fit-curve.d.ts.map +1 -1
- package/src/fit/fit-curve.js +16 -381
- package/src/fit/fit-data.d.ts +2 -1
- package/src/fit/fit-data.d.ts.map +1 -1
- package/src/fit/fit-data.js +3 -2
- package/src/fit/new-fit-API.d.ts +37 -7
- package/src/fit/new-fit-API.d.ts.map +1 -1
- package/src/fit/new-fit-API.js +279 -9
- package/src/mpo/mpo-line-editor.d.ts +10 -0
- package/src/mpo/mpo-line-editor.d.ts.map +1 -0
- package/src/mpo/mpo-line-editor.js +258 -0
- package/src/mpo/mpo-profile-editor.d.ts +13 -0
- package/src/mpo/mpo-profile-editor.d.ts.map +1 -0
- package/src/mpo/mpo-profile-editor.js +79 -0
- package/src/mpo/mpo.d.ts +19 -0
- package/src/mpo/mpo.d.ts.map +1 -0
- package/src/mpo/mpo.js +52 -0
|
@@ -0,0 +1,258 @@
|
|
|
1
|
+
/* eslint-disable max-len */
|
|
2
|
+
import * as grok from 'datagrok-api/grok';
|
|
3
|
+
import * as ui from 'datagrok-api/ui';
|
|
4
|
+
import Konva from 'konva';
|
|
5
|
+
import { Subject } from 'rxjs'; // Import type from mpo.ts
|
|
6
|
+
// Constants for the editor layout
|
|
7
|
+
const EDITOR_PADDING = { top: 10, right: 10, bottom: 20, left: 30 };
|
|
8
|
+
const POINT_RADIUS = 3;
|
|
9
|
+
export class MpoDesirabilityLineEditor {
|
|
10
|
+
constructor(prop, width, height) {
|
|
11
|
+
this.root = ui.div();
|
|
12
|
+
this.onChanged = new Subject();
|
|
13
|
+
this._prop = prop;
|
|
14
|
+
this.root.style.width = `${width}px`;
|
|
15
|
+
this.root.style.height = `${height}px`;
|
|
16
|
+
this.root.style.position = 'relative'; // Needed for absolute positioning of Konva stage
|
|
17
|
+
const that = this;
|
|
18
|
+
// Delay Konva initialization slightly to ensure this.container is in DOM
|
|
19
|
+
setTimeout(() => {
|
|
20
|
+
var _a, _b;
|
|
21
|
+
if (!this.root.parentElement) {
|
|
22
|
+
console.warn('Konva this.container not attached to DOM yet.');
|
|
23
|
+
// Optionally, retry or handle error
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
const stage = new Konva.Stage({
|
|
27
|
+
container: this.root,
|
|
28
|
+
width: width,
|
|
29
|
+
height: height,
|
|
30
|
+
});
|
|
31
|
+
const layer = new Konva.Layer();
|
|
32
|
+
stage.add(layer);
|
|
33
|
+
const minX = (_a = prop.min) !== null && _a !== void 0 ? _a : Math.min(...prop.line.map((p) => p[0]));
|
|
34
|
+
const maxX = (_b = prop.max) !== null && _b !== void 0 ? _b : Math.max(...prop.line.map((p) => p[0]));
|
|
35
|
+
// --- Draw Axes ---
|
|
36
|
+
const xAxis = new Konva.Line({
|
|
37
|
+
points: [EDITOR_PADDING.left, height - EDITOR_PADDING.bottom, width - EDITOR_PADDING.right, height - EDITOR_PADDING.bottom],
|
|
38
|
+
stroke: 'grey',
|
|
39
|
+
strokeWidth: 1,
|
|
40
|
+
});
|
|
41
|
+
const yAxis = new Konva.Line({
|
|
42
|
+
points: [EDITOR_PADDING.left, EDITOR_PADDING.top, EDITOR_PADDING.left, height - EDITOR_PADDING.bottom],
|
|
43
|
+
stroke: 'grey',
|
|
44
|
+
strokeWidth: 1,
|
|
45
|
+
});
|
|
46
|
+
layer.add(xAxis, yAxis);
|
|
47
|
+
// --- Draw Axis Labels/Ticks (Simplified) ---
|
|
48
|
+
const minXLabel = new Konva.Text({
|
|
49
|
+
x: EDITOR_PADDING.left,
|
|
50
|
+
y: height - EDITOR_PADDING.bottom + 3,
|
|
51
|
+
text: minX.toFixed(1),
|
|
52
|
+
fontSize: 9,
|
|
53
|
+
fill: 'grey',
|
|
54
|
+
});
|
|
55
|
+
const maxXLabel = new Konva.Text({
|
|
56
|
+
x: width - EDITOR_PADDING.right - 15,
|
|
57
|
+
y: height - EDITOR_PADDING.bottom + 3,
|
|
58
|
+
text: maxX.toFixed(1),
|
|
59
|
+
fontSize: 9,
|
|
60
|
+
align: 'right',
|
|
61
|
+
fill: 'grey',
|
|
62
|
+
});
|
|
63
|
+
const zeroYLabel = new Konva.Text({
|
|
64
|
+
x: EDITOR_PADDING.left - 20,
|
|
65
|
+
y: height - EDITOR_PADDING.bottom - 5,
|
|
66
|
+
text: '0.0',
|
|
67
|
+
fontSize: 9,
|
|
68
|
+
fill: 'grey',
|
|
69
|
+
});
|
|
70
|
+
const oneYLabel = new Konva.Text({
|
|
71
|
+
x: EDITOR_PADDING.left - 20,
|
|
72
|
+
y: EDITOR_PADDING.top - 5,
|
|
73
|
+
text: '1.0',
|
|
74
|
+
fontSize: 9,
|
|
75
|
+
fill: 'grey',
|
|
76
|
+
});
|
|
77
|
+
layer.add(minXLabel, maxXLabel, zeroYLabel, oneYLabel);
|
|
78
|
+
// --- Draw Line and Points ---
|
|
79
|
+
const konvaLine = new Konva.Line({
|
|
80
|
+
points: [],
|
|
81
|
+
stroke: '#2077b4',
|
|
82
|
+
strokeWidth: 2,
|
|
83
|
+
lineCap: 'round',
|
|
84
|
+
lineJoin: 'round',
|
|
85
|
+
});
|
|
86
|
+
layer.add(konvaLine);
|
|
87
|
+
const pointsGroup = new Konva.Group(); // Group for draggable points
|
|
88
|
+
layer.add(pointsGroup);
|
|
89
|
+
// Function to redraw everything based on prop.line
|
|
90
|
+
function redraw(notify = true) {
|
|
91
|
+
pointsGroup.destroyChildren(); // Clear old points
|
|
92
|
+
const konvaPoints = [];
|
|
93
|
+
prop.line.sort((a, b) => a[0] - b[0]); // Ensure sorted
|
|
94
|
+
prop.line.forEach((p, index) => {
|
|
95
|
+
const coords = toCanvasCoords(p[0], p[1], minX, maxX, width, height);
|
|
96
|
+
konvaPoints.push(coords.x, coords.y);
|
|
97
|
+
const pointCircle = new Konva.Circle({
|
|
98
|
+
x: coords.x,
|
|
99
|
+
y: coords.y,
|
|
100
|
+
radius: POINT_RADIUS,
|
|
101
|
+
fill: '#d72f30',
|
|
102
|
+
stroke: 'black',
|
|
103
|
+
strokeWidth: 1,
|
|
104
|
+
draggable: true,
|
|
105
|
+
hitStrokeWidth: 5, // Easier to hit for dragging/clicking
|
|
106
|
+
});
|
|
107
|
+
// Store index directly on the node for easy access
|
|
108
|
+
pointCircle.setAttr('_pointIndex', index);
|
|
109
|
+
// --- Dragging Logic ---
|
|
110
|
+
pointCircle.on('dragmove', (evt) => {
|
|
111
|
+
const circle = evt.target;
|
|
112
|
+
const pos = circle.position();
|
|
113
|
+
const currentPointIndex = circle.getAttr('_pointIndex');
|
|
114
|
+
// Constrain dragging horizontally between neighbors (or bounds)
|
|
115
|
+
const prevX = currentPointIndex > 0 ? prop.line[currentPointIndex - 1][0] : minX;
|
|
116
|
+
const nextX = currentPointIndex < prop.line.length - 1 ? prop.line[currentPointIndex + 1][0] : maxX;
|
|
117
|
+
// Add a small buffer to avoid points overlapping exactly
|
|
118
|
+
const buffer = (maxX - minX === 0) ? 0 : 0.001 * (maxX - minX); // Avoid NaN if minX === maxX
|
|
119
|
+
const minCanvasX = toCanvasCoords(prevX + (currentPointIndex > 0 ? buffer : 0), 0, minX, maxX, width, height).x;
|
|
120
|
+
const maxCanvasX = toCanvasCoords(nextX - (currentPointIndex < prop.line.length - 1 ? buffer : 0), 0, minX, maxX, width, height).x;
|
|
121
|
+
pos.x = Math.max(minCanvasX, Math.min(maxCanvasX, pos.x));
|
|
122
|
+
// Constrain dragging vertically
|
|
123
|
+
const plotTop = EDITOR_PADDING.top;
|
|
124
|
+
const plotBottom = height - EDITOR_PADDING.bottom;
|
|
125
|
+
pos.y = Math.max(plotTop, Math.min(plotBottom, pos.y));
|
|
126
|
+
circle.position(pos); // Update position after constraints
|
|
127
|
+
// Update data array
|
|
128
|
+
const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);
|
|
129
|
+
prop.line[currentPointIndex][0] = dataCoords.x;
|
|
130
|
+
prop.line[currentPointIndex][1] = dataCoords.y;
|
|
131
|
+
// Update the connecting line during drag
|
|
132
|
+
const currentKonvaPoints = prop.line.map((pData, idx) => {
|
|
133
|
+
// Use dragged circle position directly for the point being dragged
|
|
134
|
+
if (idx === currentPointIndex)
|
|
135
|
+
return [pos.x, pos.y];
|
|
136
|
+
else {
|
|
137
|
+
const c = toCanvasCoords(pData[0], pData[1], minX, maxX, width, height);
|
|
138
|
+
return [c.x, c.y];
|
|
139
|
+
}
|
|
140
|
+
}).flat();
|
|
141
|
+
konvaLine.points(currentKonvaPoints);
|
|
142
|
+
layer.batchDraw(); // More efficient redraw
|
|
143
|
+
});
|
|
144
|
+
pointCircle.on('dragend', (evt) => {
|
|
145
|
+
// Ensure data is sorted after drag, although constraints should handle it
|
|
146
|
+
prop.line.sort((a, b) => a[0] - b[0]);
|
|
147
|
+
redraw(); // Full redraw on drag end to fix indices and line path
|
|
148
|
+
});
|
|
149
|
+
// --- Right-click to Remove ---
|
|
150
|
+
pointCircle.on('contextmenu', (evt) => {
|
|
151
|
+
evt.evt.preventDefault(); // Prevent browser context menu
|
|
152
|
+
// Keep at least 2 points for a line segment
|
|
153
|
+
if (prop.line.length <= 2) {
|
|
154
|
+
// Use DG tooltip or simple alert/warning
|
|
155
|
+
grok.shell.warning('Cannot remove points, minimum of 2 required.');
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
const circle = evt.target;
|
|
159
|
+
const indexToRemove = circle.getAttr('_pointIndex');
|
|
160
|
+
prop.line.splice(indexToRemove, 1);
|
|
161
|
+
redraw(); // Redraw after removal
|
|
162
|
+
});
|
|
163
|
+
// Enhance usability: change cursor on hover and show tooltip
|
|
164
|
+
pointCircle.on('mouseenter', (evt) => {
|
|
165
|
+
stage.container().style.cursor = 'pointer';
|
|
166
|
+
const circle = evt.target;
|
|
167
|
+
const pos = circle.position();
|
|
168
|
+
const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);
|
|
169
|
+
const tooltipText = `X: ${dataCoords.x.toFixed(2)}, Y: ${dataCoords.y.toFixed(2)}<br><br>Drag to move, right-click to delete`;
|
|
170
|
+
ui.tooltip.show(tooltipText, evt.evt.clientX, evt.evt.clientY);
|
|
171
|
+
});
|
|
172
|
+
pointCircle.on('mouseleave', (evt) => {
|
|
173
|
+
stage.container().style.cursor = 'default';
|
|
174
|
+
ui.tooltip.hide();
|
|
175
|
+
});
|
|
176
|
+
pointsGroup.add(pointCircle);
|
|
177
|
+
});
|
|
178
|
+
konvaLine.points(konvaPoints);
|
|
179
|
+
layer.batchDraw();
|
|
180
|
+
if (notify)
|
|
181
|
+
that.onChanged.next();
|
|
182
|
+
}
|
|
183
|
+
// --- Left-click to Add Point ---
|
|
184
|
+
stage.on('click tap', (evt) => {
|
|
185
|
+
// Ignore clicks on existing points (circles) or non-left clicks
|
|
186
|
+
if (evt.target instanceof Konva.Circle || evt.evt.button !== 0)
|
|
187
|
+
return;
|
|
188
|
+
const pos = stage.getPointerPosition();
|
|
189
|
+
if (!pos)
|
|
190
|
+
return;
|
|
191
|
+
// Ensure click is within the plot area boundaries
|
|
192
|
+
if (pos.x < EDITOR_PADDING.left || pos.x > width - EDITOR_PADDING.right ||
|
|
193
|
+
pos.y < EDITOR_PADDING.top || pos.y > height - EDITOR_PADDING.bottom)
|
|
194
|
+
return;
|
|
195
|
+
const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);
|
|
196
|
+
// Add the new point
|
|
197
|
+
prop.line.push([dataCoords.x, dataCoords.y]);
|
|
198
|
+
// No need to sort here, redraw() handles sorting
|
|
199
|
+
redraw();
|
|
200
|
+
});
|
|
201
|
+
// Change cursor when over the stage plot area for adding points
|
|
202
|
+
stage.on('mouseenter', (evt) => {
|
|
203
|
+
if (!(evt.target instanceof Konva.Circle)) {
|
|
204
|
+
const pos = stage.getPointerPosition();
|
|
205
|
+
if (pos && pos.x >= EDITOR_PADDING.left && pos.x <= width - EDITOR_PADDING.right &&
|
|
206
|
+
pos.y >= EDITOR_PADDING.top && pos.y <= height - EDITOR_PADDING.bottom)
|
|
207
|
+
stage.container().style.cursor = 'crosshair';
|
|
208
|
+
}
|
|
209
|
+
});
|
|
210
|
+
stage.on('mouseleave', (evt) => {
|
|
211
|
+
if (!(evt.target instanceof Konva.Circle))
|
|
212
|
+
stage.container().style.cursor = 'default';
|
|
213
|
+
});
|
|
214
|
+
stage.on('mousemove', (evt) => {
|
|
215
|
+
if (!(evt.target instanceof Konva.Circle)) {
|
|
216
|
+
const pos = stage.getPointerPosition();
|
|
217
|
+
if (pos && pos.x >= EDITOR_PADDING.left && pos.x <= width - EDITOR_PADDING.right &&
|
|
218
|
+
pos.y >= EDITOR_PADDING.top && pos.y <= height - EDITOR_PADDING.bottom)
|
|
219
|
+
stage.container().style.cursor = 'crosshair';
|
|
220
|
+
else
|
|
221
|
+
stage.container().style.cursor = 'default';
|
|
222
|
+
}
|
|
223
|
+
});
|
|
224
|
+
stage.on('mouseout', (_) => ui.tooltip.hide());
|
|
225
|
+
// Initial draw
|
|
226
|
+
redraw(false);
|
|
227
|
+
}, 0);
|
|
228
|
+
}
|
|
229
|
+
get line() {
|
|
230
|
+
return this._prop.line;
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
// Function to transform data coordinates to canvas coordinates
|
|
234
|
+
function toCanvasCoords(x, y, minX, maxX, width, height) {
|
|
235
|
+
const plotWidth = width - EDITOR_PADDING.left - EDITOR_PADDING.right;
|
|
236
|
+
const plotHeight = height - EDITOR_PADDING.top - EDITOR_PADDING.bottom;
|
|
237
|
+
// Handle case where minX === maxX to avoid division by zero
|
|
238
|
+
const scaleX = (maxX - minX === 0) ? 1 : plotWidth / (maxX - minX);
|
|
239
|
+
const scaleY = plotHeight; // y data is 0-1
|
|
240
|
+
const canvasX = EDITOR_PADDING.left + (x - minX) * scaleX;
|
|
241
|
+
const canvasY = EDITOR_PADDING.top + plotHeight - (y * scaleY); // Flip y-axis
|
|
242
|
+
return { x: canvasX, y: canvasY };
|
|
243
|
+
}
|
|
244
|
+
// Function to transform canvas coordinates to data coordinates
|
|
245
|
+
function toDataCoords(canvasX, canvasY, minX, maxX, width, height) {
|
|
246
|
+
const plotWidth = width - EDITOR_PADDING.left - EDITOR_PADDING.right;
|
|
247
|
+
const plotHeight = height - EDITOR_PADDING.top - EDITOR_PADDING.bottom;
|
|
248
|
+
// Handle case where minX === maxX
|
|
249
|
+
const scaleX = (maxX - minX === 0) ? 1 : plotWidth / (maxX - minX);
|
|
250
|
+
const scaleY = plotHeight;
|
|
251
|
+
let dataX = minX + (canvasX - EDITOR_PADDING.left) / scaleX;
|
|
252
|
+
let dataY = (EDITOR_PADDING.top + plotHeight - canvasY) / scaleY;
|
|
253
|
+
// Clamp values
|
|
254
|
+
dataX = Math.max(minX, Math.min(maxX, dataX));
|
|
255
|
+
dataY = Math.max(0, Math.min(1, dataY));
|
|
256
|
+
return { x: dataX, y: dataY };
|
|
257
|
+
}
|
|
258
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mpo-line-editor.js","sourceRoot":"","sources":["mpo-line-editor.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable max-len */\nimport * as grok from 'datagrok-api/grok';\nimport * as ui from 'datagrok-api/ui';\n\nimport Konva from 'konva';\nimport {DesirabilityLine, PropertyDesirability} from './mpo';\nimport {Subject} from 'rxjs'; // Import type from mpo.ts\n\n// Constants for the editor layout\nconst EDITOR_PADDING = {top: 10, right: 10, bottom: 20, left: 30};\nconst POINT_RADIUS = 3;\n\n\nexport class MpoDesirabilityLineEditor {\n  root = ui.div();\n  onChanged = new Subject();\n  private _prop: PropertyDesirability;\n\n  constructor(prop: PropertyDesirability, width: number, height: number) {\n    this._prop = prop;\n    this.root.style.width = `${width}px`;\n    this.root.style.height = `${height}px`;\n    this.root.style.position = 'relative'; // Needed for absolute positioning of Konva stage\n    const that = this;\n\n    // Delay Konva initialization slightly to ensure this.container is in DOM\n    setTimeout(() => {\n      if (!this.root.parentElement) {\n        console.warn('Konva this.container not attached to DOM yet.');\n        // Optionally, retry or handle error\n        return;\n      }\n\n      const stage = new Konva.Stage({\n        container: this.root, // Use the this.container div\n        width: width,\n        height: height,\n      });\n\n      const layer = new Konva.Layer();\n      stage.add(layer);\n\n      const minX = prop.min ?? Math.min(...prop.line.map((p) => p[0]));\n      const maxX = prop.max ?? Math.max(...prop.line.map((p) => p[0]));\n\n      // --- Draw Axes ---\n      const xAxis = new Konva.Line({\n        points: [EDITOR_PADDING.left, height - EDITOR_PADDING.bottom, width - EDITOR_PADDING.right, height - EDITOR_PADDING.bottom],\n        stroke: 'grey', // Lighter color for axes\n        strokeWidth: 1,\n      });\n      const yAxis = new Konva.Line({\n        points: [EDITOR_PADDING.left, EDITOR_PADDING.top, EDITOR_PADDING.left, height - EDITOR_PADDING.bottom],\n        stroke: 'grey', // Lighter color for axes\n        strokeWidth: 1,\n      });\n      layer.add(xAxis, yAxis);\n\n      // --- Draw Axis Labels/Ticks (Simplified) ---\n      const minXLabel = new Konva.Text({\n        x: EDITOR_PADDING.left,\n        y: height - EDITOR_PADDING.bottom + 3, // Below axis\n        text: minX.toFixed(1), // Fewer decimals for smaller space\n        fontSize: 9,\n        fill: 'grey',\n      });\n      const maxXLabel = new Konva.Text({\n        x: width - EDITOR_PADDING.right - 15, // Adjust position\n        y: height - EDITOR_PADDING.bottom + 3, // Below axis\n        text: maxX.toFixed(1), // Fewer decimals\n        fontSize: 9,\n        align: 'right',\n        fill: 'grey',\n      });\n      const zeroYLabel = new Konva.Text({\n        x: EDITOR_PADDING.left - 20, // Left of axis\n        y: height - EDITOR_PADDING.bottom - 5, // Align with axis bottom\n        text: '0.0',\n        fontSize: 9,\n        fill: 'grey',\n      });\n      const oneYLabel = new Konva.Text({\n        x: EDITOR_PADDING.left - 20, // Left of axis\n        y: EDITOR_PADDING.top - 5, // Align with axis top\n        text: '1.0',\n        fontSize: 9,\n        fill: 'grey',\n      });\n      layer.add(minXLabel, maxXLabel, zeroYLabel, oneYLabel);\n\n\n      // --- Draw Line and Points ---\n      const konvaLine = new Konva.Line({\n        points: [], // Will be populated by points\n        stroke: '#2077b4',\n        strokeWidth: 2,\n        lineCap: 'round',\n        lineJoin: 'round',\n      });\n      layer.add(konvaLine);\n\n      const pointsGroup = new Konva.Group(); // Group for draggable points\n      layer.add(pointsGroup);\n\n      // Function to redraw everything based on prop.line\n      function redraw(notify: boolean = true) {\n        pointsGroup.destroyChildren(); // Clear old points\n        const konvaPoints: number[] = [];\n\n        prop.line.sort((a, b) => a[0] - b[0]); // Ensure sorted\n\n        prop.line.forEach((p, index) => {\n          const coords = toCanvasCoords(p[0], p[1], minX, maxX, width, height);\n          konvaPoints.push(coords.x, coords.y);\n\n          const pointCircle = new Konva.Circle({\n            x: coords.x,\n            y: coords.y,\n            radius: POINT_RADIUS,\n            fill: '#d72f30',\n            stroke: 'black',\n            strokeWidth: 1,\n            draggable: true, // Make points draggable\n            hitStrokeWidth: 5, // Easier to hit for dragging/clicking\n          });\n          // Store index directly on the node for easy access\n          pointCircle.setAttr('_pointIndex', index);\n\n          // --- Dragging Logic ---\n          pointCircle.on('dragmove', (evt: Konva.KonvaEventObject<DragEvent>) => {\n            const circle = evt.target as Konva.Circle;\n            const pos = circle.position();\n            const currentPointIndex = circle.getAttr('_pointIndex');\n\n            // Constrain dragging horizontally between neighbors (or bounds)\n            const prevX = currentPointIndex > 0 ? prop.line[currentPointIndex - 1][0] : minX;\n            const nextX = currentPointIndex < prop.line.length - 1 ? prop.line[currentPointIndex + 1][0] : maxX;\n            // Add a small buffer to avoid points overlapping exactly\n            const buffer = (maxX - minX === 0) ? 0 : 0.001 * (maxX - minX); // Avoid NaN if minX === maxX\n            const minCanvasX = toCanvasCoords(prevX + (currentPointIndex > 0 ? buffer : 0), 0, minX, maxX, width, height).x;\n            const maxCanvasX = toCanvasCoords(nextX - (currentPointIndex < prop.line.length - 1 ? buffer : 0), 0, minX, maxX, width, height).x;\n\n            pos.x = Math.max(minCanvasX, Math.min(maxCanvasX, pos.x));\n\n            // Constrain dragging vertically\n            const plotTop = EDITOR_PADDING.top;\n            const plotBottom = height - EDITOR_PADDING.bottom;\n            pos.y = Math.max(plotTop, Math.min(plotBottom, pos.y));\n\n            circle.position(pos); // Update position after constraints\n\n            // Update data array\n            const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);\n            prop.line[currentPointIndex][0] = dataCoords.x;\n            prop.line[currentPointIndex][1] = dataCoords.y;\n\n            // Update the connecting line during drag\n            const currentKonvaPoints = prop.line.map((pData, idx) => {\n              // Use dragged circle position directly for the point being dragged\n              if (idx === currentPointIndex)\n                return [pos.x, pos.y];\n              else {\n                const c = toCanvasCoords(pData[0], pData[1], minX, maxX, width, height);\n                return [c.x, c.y];\n              }\n            }).flat();\n            konvaLine.points(currentKonvaPoints);\n            layer.batchDraw(); // More efficient redraw\n          });\n\n          pointCircle.on('dragend', (evt: Konva.KonvaEventObject<DragEvent>) => {\n            // Ensure data is sorted after drag, although constraints should handle it\n            prop.line.sort((a, b) => a[0] - b[0]);\n            redraw(); // Full redraw on drag end to fix indices and line path\n          });\n\n          // --- Right-click to Remove ---\n          pointCircle.on('contextmenu', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n            evt.evt.preventDefault(); // Prevent browser context menu\n            // Keep at least 2 points for a line segment\n            if (prop.line.length <= 2) {\n              // Use DG tooltip or simple alert/warning\n              grok.shell.warning('Cannot remove points, minimum of 2 required.');\n              return;\n            }\n\n            const circle = evt.target as Konva.Circle;\n            const indexToRemove = circle.getAttr('_pointIndex');\n            prop.line.splice(indexToRemove, 1);\n            redraw(); // Redraw after removal\n          });\n\n          // Enhance usability: change cursor on hover and show tooltip\n          pointCircle.on('mouseenter', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n            stage.container().style.cursor = 'pointer';\n            const circle = evt.target as Konva.Circle;\n            const pos = circle.position();\n            const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);\n            const tooltipText = `X: ${dataCoords.x.toFixed(2)}, Y: ${dataCoords.y.toFixed(2)}<br><br>Drag to move, right-click to delete`;\n            ui.tooltip.show(tooltipText, evt.evt.clientX, evt.evt.clientY);\n          });\n          pointCircle.on('mouseleave', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n            stage.container().style.cursor = 'default';\n            ui.tooltip.hide();\n          });\n\n          pointsGroup.add(pointCircle);\n        });\n\n        konvaLine.points(konvaPoints);\n        layer.batchDraw();\n\n        if (notify)\n          that.onChanged.next();\n      }\n\n      // --- Left-click to Add Point ---\n      stage.on('click tap', (evt: Konva.KonvaEventObject<PointerEvent>) => {\n        // Ignore clicks on existing points (circles) or non-left clicks\n        if (evt.target instanceof Konva.Circle || evt.evt.button !== 0) return;\n\n        const pos = stage.getPointerPosition();\n        if (!pos) return;\n\n        // Ensure click is within the plot area boundaries\n        if (pos.x < EDITOR_PADDING.left || pos.x > width - EDITOR_PADDING.right ||\n          pos.y < EDITOR_PADDING.top || pos.y > height - EDITOR_PADDING.bottom)\n          return;\n\n        const dataCoords = toDataCoords(pos.x, pos.y, minX, maxX, width, height);\n        // Add the new point\n        prop.line.push([dataCoords.x, dataCoords.y]);\n\n        // No need to sort here, redraw() handles sorting\n        redraw();\n      });\n\n      // Change cursor when over the stage plot area for adding points\n      stage.on('mouseenter', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n        if (!(evt.target instanceof Konva.Circle)) {\n          const pos = stage.getPointerPosition();\n          if (pos && pos.x >= EDITOR_PADDING.left && pos.x <= width - EDITOR_PADDING.right &&\n            pos.y >= EDITOR_PADDING.top && pos.y <= height - EDITOR_PADDING.bottom)\n            stage.container().style.cursor = 'crosshair';\n        }\n      });\n\n      stage.on('mouseleave', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n        if (!(evt.target instanceof Konva.Circle))\n          stage.container().style.cursor = 'default';\n      });\n\n      stage.on('mousemove', (evt: Konva.KonvaEventObject<MouseEvent>) => {\n        if (!(evt.target instanceof Konva.Circle)) {\n          const pos = stage.getPointerPosition();\n          if (pos && pos.x >= EDITOR_PADDING.left && pos.x <= width - EDITOR_PADDING.right &&\n            pos.y >= EDITOR_PADDING.top && pos.y <= height - EDITOR_PADDING.bottom)\n            stage.container().style.cursor = 'crosshair';\n\n          else\n            stage.container().style.cursor = 'default';\n        }\n      });\n\n      stage.on('mouseout', (_) => ui.tooltip.hide());\n\n      // Initial draw\n      redraw(false);\n    }, 0);\n  }\n\n  get line(): DesirabilityLine {\n    return this._prop.line;\n  }\n}\n\n\n// Function to transform data coordinates to canvas coordinates\nfunction toCanvasCoords(x: number, y: number, minX: number, maxX: number, width: number, height: number): { x: number, y: number } {\n  const plotWidth = width - EDITOR_PADDING.left - EDITOR_PADDING.right;\n  const plotHeight = height - EDITOR_PADDING.top - EDITOR_PADDING.bottom;\n  // Handle case where minX === maxX to avoid division by zero\n  const scaleX = (maxX - minX === 0) ? 1 : plotWidth / (maxX - minX);\n  const scaleY = plotHeight; // y data is 0-1\n\n  const canvasX = EDITOR_PADDING.left + (x - minX) * scaleX;\n  const canvasY = EDITOR_PADDING.top + plotHeight - (y * scaleY); // Flip y-axis\n\n  return {x: canvasX, y: canvasY};\n}\n\n// Function to transform canvas coordinates to data coordinates\nfunction toDataCoords(canvasX: number, canvasY: number, minX: number, maxX: number, width: number, height: number): { x: number, y: number } {\n  const plotWidth = width - EDITOR_PADDING.left - EDITOR_PADDING.right;\n  const plotHeight = height - EDITOR_PADDING.top - EDITOR_PADDING.bottom;\n  // Handle case where minX === maxX\n  const scaleX = (maxX - minX === 0) ? 1 : plotWidth / (maxX - minX);\n  const scaleY = plotHeight;\n\n  let dataX = minX + (canvasX - EDITOR_PADDING.left) / scaleX;\n  let dataY = (EDITOR_PADDING.top + plotHeight - canvasY) / scaleY;\n\n  // Clamp values\n  dataX = Math.max(minX, Math.min(maxX, dataX));\n  dataY = Math.max(0, Math.min(1, dataY));\n\n  return {x: dataX, y: dataY};\n}\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import * as DG from 'datagrok-api/dg';
|
|
2
|
+
import { DesirabilityProfile } from './mpo';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
export declare class MpoProfileEditor {
|
|
5
|
+
root: HTMLDivElement;
|
|
6
|
+
dataFrame?: DG.DataFrame;
|
|
7
|
+
onChanged: Subject<unknown>;
|
|
8
|
+
profile?: DesirabilityProfile;
|
|
9
|
+
constructor(dataFrame?: DG.DataFrame);
|
|
10
|
+
getProfile(): DesirabilityProfile | undefined;
|
|
11
|
+
setProfile(profile?: DesirabilityProfile): void;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=mpo-profile-editor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mpo-profile-editor.d.ts","sourceRoot":"","sources":["mpo-profile-editor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAGtC,OAAO,EAAC,mBAAmB,EAAC,MAAM,OAAO,CAAC;AAE1C,OAAO,EAAC,OAAO,EAAC,MAAM,MAAM,CAAC;AAG7B,qBAAa,gBAAgB;IAC3B,IAAI,iBAAc;IAClB,SAAS,CAAC,EAAE,EAAE,CAAC,SAAS,CAAC;IACzB,SAAS,mBAAiB;IAC1B,OAAO,CAAC,EAAE,mBAAmB,CAAC;gBAElB,SAAS,CAAC,EAAE,EAAE,CAAC,SAAS;IAKpC,UAAU,IAAI,mBAAmB,GAAG,SAAS;IAI7C,UAAU,CAAC,OAAO,CAAC,EAAE,mBAAmB;CAsEzC"}
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import * as ui from 'datagrok-api/ui';
|
|
2
|
+
import { MpoDesirabilityLineEditor } from './mpo-line-editor';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
export class MpoProfileEditor {
|
|
5
|
+
constructor(dataFrame) {
|
|
6
|
+
this.root = ui.div([]);
|
|
7
|
+
this.onChanged = new Subject();
|
|
8
|
+
this.dataFrame = dataFrame;
|
|
9
|
+
this.setProfile();
|
|
10
|
+
}
|
|
11
|
+
getProfile() {
|
|
12
|
+
return this.profile;
|
|
13
|
+
}
|
|
14
|
+
setProfile(profile) {
|
|
15
|
+
this.profile = profile;
|
|
16
|
+
ui.empty(this.root);
|
|
17
|
+
if (!profile) {
|
|
18
|
+
this.root.append(ui.divText('No profile specified.'));
|
|
19
|
+
return;
|
|
20
|
+
}
|
|
21
|
+
// Create header row for the table
|
|
22
|
+
const header = ui.divH([
|
|
23
|
+
ui.divText('Property', { style: { fontWeight: 'bold', width: '150px' } }),
|
|
24
|
+
ui.divText('Weight', { style: { fontWeight: 'bold', width: '60px' } }),
|
|
25
|
+
ui.divText('Desirability', { style: { fontWeight: 'bold', flexGrow: '1' } }), // Let editor take space
|
|
26
|
+
], { style: {
|
|
27
|
+
marginTop: '10px',
|
|
28
|
+
paddingBottom: '5px',
|
|
29
|
+
borderBottom: '1px solid #ccc',
|
|
30
|
+
},
|
|
31
|
+
});
|
|
32
|
+
const propertyRows = Object.entries(profile.properties).map(([propertyName, prop]) => {
|
|
33
|
+
var _a, _b, _c;
|
|
34
|
+
const lineEditor = new MpoDesirabilityLineEditor(prop, 300, 80);
|
|
35
|
+
lineEditor.onChanged.subscribe((_) => this.onChanged.next());
|
|
36
|
+
// Input for weight - updates the *copy* of the template data
|
|
37
|
+
const weightInput = ui.input.float('', {
|
|
38
|
+
value: prop.weight, min: 0, max: 1,
|
|
39
|
+
onValueChanged: (newValue) => {
|
|
40
|
+
if (profile && profile.properties[propertyName]) {
|
|
41
|
+
// Update the weight in the temporary template object
|
|
42
|
+
let clampedWeight = newValue !== null && newValue !== void 0 ? newValue : 0;
|
|
43
|
+
clampedWeight = Math.max(0, Math.min(1, clampedWeight)); // Clamp 0-1
|
|
44
|
+
profile.properties[propertyName].weight = clampedWeight;
|
|
45
|
+
}
|
|
46
|
+
},
|
|
47
|
+
});
|
|
48
|
+
weightInput.root.style.width = '60px';
|
|
49
|
+
weightInput.root.style.marginTop = '21px';
|
|
50
|
+
const matchedColumnName = this.dataFrame ?
|
|
51
|
+
this.dataFrame.columns.names().find((name) => name.toLowerCase() == propertyName.toLowerCase()) :
|
|
52
|
+
null;
|
|
53
|
+
const columnInput = ui.input.choice('', {
|
|
54
|
+
nullable: true,
|
|
55
|
+
items: (_c = (_b = (_a = this.dataFrame) === null || _a === void 0 ? void 0 : _a.columns) === null || _b === void 0 ? void 0 : _b.names()) !== null && _c !== void 0 ? _c : [''],
|
|
56
|
+
value: matchedColumnName !== null && matchedColumnName !== void 0 ? matchedColumnName : ''
|
|
57
|
+
});
|
|
58
|
+
const rowDiv = ui.divH([
|
|
59
|
+
ui.divV([
|
|
60
|
+
ui.divText(propertyName, { style: { width: '150px', paddingTop: '5px', marginLeft: '4px' } }),
|
|
61
|
+
this.dataFrame ? columnInput.root : null,
|
|
62
|
+
]),
|
|
63
|
+
weightInput.root,
|
|
64
|
+
lineEditor.root, // Add the Konva container div
|
|
65
|
+
]);
|
|
66
|
+
rowDiv.style.alignItems = 'center'; // Vertically align items in the row
|
|
67
|
+
rowDiv.style.marginBottom = '5px'; // Space between rows
|
|
68
|
+
rowDiv.style.minHeight = '70px'; // Ensure consistent row height even if editor takes time
|
|
69
|
+
return rowDiv;
|
|
70
|
+
}).filter((el) => el !== null); // Filter out skipped properties
|
|
71
|
+
if (propertyRows.length > 0) {
|
|
72
|
+
this.root.append(header);
|
|
73
|
+
this.root.append(ui.divV(propertyRows)); // Cast needed after filter
|
|
74
|
+
}
|
|
75
|
+
else
|
|
76
|
+
this.root.append(ui.divText('No matching properties found in the table for this template.'));
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mpo-profile-editor.js","sourceRoot":"","sources":["mpo-profile-editor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as DG from 'datagrok-api/dg';\nimport * as ui from 'datagrok-api/ui';\n\nimport {DesirabilityProfile} from './mpo';\nimport {MpoDesirabilityLineEditor} from './mpo-line-editor';\nimport {Subject} from 'rxjs';\n\n\nexport class MpoProfileEditor {\n  root = ui.div([]);\n  dataFrame?: DG.DataFrame;\n  onChanged = new Subject();\n  profile?: DesirabilityProfile;\n\n  constructor(dataFrame?: DG.DataFrame) {\n    this.dataFrame = dataFrame;\n    this.setProfile();\n  }\n\n  getProfile(): DesirabilityProfile | undefined {\n    return this.profile;\n  }\n\n  setProfile(profile?: DesirabilityProfile) {\n    this.profile = profile;\n    ui.empty(this.root);\n    if (!profile) {\n      this.root.append(ui.divText('No profile specified.'));\n      return;\n    }\n\n    // Create header row for the table\n    const header = ui.divH([\n      ui.divText('Property', {style: {fontWeight: 'bold', width: '150px'}}),\n      ui.divText('Weight', {style: {fontWeight: 'bold', width: '60px'}}),\n      ui.divText('Desirability', {style: {fontWeight: 'bold', flexGrow: '1'}}), // Let editor take space\n    ], {style: {\n      marginTop: '10px',\n      paddingBottom: '5px',\n      borderBottom: '1px solid #ccc',\n    },\n    });\n\n    const propertyRows = Object.entries(profile.properties).map(([propertyName, prop]) => {\n      const lineEditor = new MpoDesirabilityLineEditor(prop, 300, 80);\n      lineEditor.onChanged.subscribe((_) => this.onChanged.next());\n\n      // Input for weight - updates the *copy* of the template data\n      const weightInput = ui.input.float('', { // No label needed here\n        value: prop.weight, min: 0, max: 1,\n        onValueChanged: (newValue) => { // Changed parameter name for clarity\n          if (profile && profile.properties[propertyName]) {\n            // Update the weight in the temporary template object\n            let clampedWeight = newValue ?? 0;\n            clampedWeight = Math.max(0, Math.min(1, clampedWeight)); // Clamp 0-1\n            profile.properties[propertyName].weight = clampedWeight;\n          }\n        },\n      });\n\n      weightInput.root.style.width = '60px';\n      weightInput.root.style.marginTop = '21px';\n\n      const matchedColumnName = this.dataFrame ?\n        this.dataFrame.columns.names().find((name) => name.toLowerCase() == propertyName.toLowerCase()) :\n        null;\n\n      const columnInput = ui.input.choice('', {\n        nullable: true,\n        items: this.dataFrame?.columns?.names() ?? [''],\n        value: matchedColumnName ?? ''});\n\n      const rowDiv = ui.divH([\n        ui.divV([\n          ui.divText(propertyName, {style: {width: '150px', paddingTop: '5px', marginLeft: '4px'}}),\n          this.dataFrame ? columnInput.root : null,\n        ]),\n        weightInput.root,\n        lineEditor.root, // Add the Konva container div\n      ]);\n      rowDiv.style.alignItems = 'center'; // Vertically align items in the row\n      rowDiv.style.marginBottom = '5px'; // Space between rows\n      rowDiv.style.minHeight = '70px'; // Ensure consistent row height even if editor takes time\n\n      return rowDiv;\n    }).filter((el) => el !== null); // Filter out skipped properties\n\n    if (propertyRows.length > 0) {\n      this.root.append(header);\n      this.root.append(ui.divV(propertyRows as HTMLElement[])); // Cast needed after filter\n    } else\n      this.root.append(ui.divText('No matching properties found in the table for this template.'));\n  }\n}\n"]}
|
package/src/mpo/mpo.d.ts
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import * as DG from 'datagrok-api/dg';
|
|
2
|
+
export type DesirabilityLine = number[][];
|
|
3
|
+
export type PropertyDesirability = {
|
|
4
|
+
line: DesirabilityLine;
|
|
5
|
+
min?: number;
|
|
6
|
+
max?: number;
|
|
7
|
+
weight: number;
|
|
8
|
+
};
|
|
9
|
+
export type DesirabilityProfile = {
|
|
10
|
+
name: string;
|
|
11
|
+
description: string;
|
|
12
|
+
properties: {
|
|
13
|
+
[key: string]: PropertyDesirability;
|
|
14
|
+
};
|
|
15
|
+
};
|
|
16
|
+
export declare function desirabilityScore(x: number, desirabilityLine: DesirabilityLine): number;
|
|
17
|
+
/** Calculates the multi parameter optimization score, 0-100, 100 is the maximum */
|
|
18
|
+
export declare function mpo(dataFrame: DG.DataFrame, columns: DG.Column[]): DG.Column;
|
|
19
|
+
//# sourceMappingURL=mpo.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mpo.d.ts","sourceRoot":"","sources":["mpo.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAItC,MAAM,MAAM,gBAAgB,GAAG,MAAM,EAAE,EAAE,CAAC;AAG1C,MAAM,MAAM,oBAAoB,GAAG;IACjC,IAAI,EAAE,gBAAgB,CAAC;IACvB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;CAChB,CAAA;AAGD,MAAM,MAAM,mBAAmB,GAAG;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,oBAAoB,CAAA;KAAE,CAAC;CACrD,CAAA;AAKD,wBAAgB,iBAAiB,CAAC,CAAC,EAAE,MAAM,EAAE,gBAAgB,EAAE,gBAAgB,GAAG,MAAM,CAoBvF;AAED,mFAAmF;AACnF,wBAAgB,GAAG,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,EAAE,OAAO,EAAE,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE,CAAC,MAAM,CA8B5E"}
|
package/src/mpo/mpo.js
ADDED
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
/* eslint-disable guard-for-in */
|
|
2
|
+
/* eslint-disable max-len */
|
|
3
|
+
import * as DG from 'datagrok-api/dg';
|
|
4
|
+
/// Calculates the desirability score for a given x value
|
|
5
|
+
/// Returns 0 if x is outside the range of the desirability line
|
|
6
|
+
/// Otherwise, returns the y value of the desirability line at x
|
|
7
|
+
export function desirabilityScore(x, desirabilityLine) {
|
|
8
|
+
// If the line is empty or x is outside the range, return 0
|
|
9
|
+
if (desirabilityLine.length === 0 || x < desirabilityLine[0][0] || x > desirabilityLine[desirabilityLine.length - 1][0])
|
|
10
|
+
return 0;
|
|
11
|
+
// Find the two points that x lies between
|
|
12
|
+
for (let i = 0; i < desirabilityLine.length - 1; i++) {
|
|
13
|
+
const [x1, y1] = desirabilityLine[i];
|
|
14
|
+
const [x2, y2] = desirabilityLine[i + 1];
|
|
15
|
+
if (x >= x1 && x <= x2) {
|
|
16
|
+
// Linear interpolation between the two points
|
|
17
|
+
if (x1 === x2)
|
|
18
|
+
return y1;
|
|
19
|
+
const slope = (y2 - y1) / (x2 - x1);
|
|
20
|
+
return y1 + slope * (x - x1);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
// Should not happen if x is within bounds, but return 0 as fallback
|
|
24
|
+
return 0;
|
|
25
|
+
}
|
|
26
|
+
/** Calculates the multi parameter optimization score, 0-100, 100 is the maximum */
|
|
27
|
+
export function mpo(dataFrame, columns) {
|
|
28
|
+
if (columns.length === 0)
|
|
29
|
+
throw new Error('No columns provided for MPO calculation.');
|
|
30
|
+
const resultColumnName = dataFrame.columns.getUnusedName('MPO'); // Ensure unique name
|
|
31
|
+
const resultColumn = DG.Column.float(resultColumnName, columns[0].length);
|
|
32
|
+
const desirabilityTemplates = columns.map((column) => {
|
|
33
|
+
const tag = column.getTag('desirabilityTemplate');
|
|
34
|
+
return JSON.parse(tag);
|
|
35
|
+
});
|
|
36
|
+
resultColumn.init((i) => {
|
|
37
|
+
let totalScore = 0;
|
|
38
|
+
let maxScore = 0;
|
|
39
|
+
for (let j = 0; j < columns.length; j++) {
|
|
40
|
+
const desirability = desirabilityTemplates[j];
|
|
41
|
+
const value = columns[j].get(i);
|
|
42
|
+
const score = desirabilityScore(value, desirability.line);
|
|
43
|
+
totalScore += desirability.weight * score;
|
|
44
|
+
maxScore += desirability.weight;
|
|
45
|
+
}
|
|
46
|
+
return 100 * (totalScore / maxScore);
|
|
47
|
+
});
|
|
48
|
+
// Add the column to the table
|
|
49
|
+
dataFrame.columns.add(resultColumn);
|
|
50
|
+
return resultColumn;
|
|
51
|
+
}
|
|
52
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibXBvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsibXBvLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGlDQUFpQztBQUNqQyw0QkFBNEI7QUFDNUIsT0FBTyxLQUFLLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQXFCdEMseURBQXlEO0FBQ3pELGdFQUFnRTtBQUNoRSxnRUFBZ0U7QUFDaEUsTUFBTSxVQUFVLGlCQUFpQixDQUFDLENBQVMsRUFBRSxnQkFBa0M7SUFDN0UsMkRBQTJEO0lBQzNELElBQUksZ0JBQWdCLENBQUMsTUFBTSxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLGdCQUFnQixDQUFDLGdCQUFnQixDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDckgsT0FBTyxDQUFDLENBQUM7SUFFWCwwQ0FBMEM7SUFDMUMsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLGdCQUFnQixDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDcEQsTUFBTSxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsR0FBRyxnQkFBZ0IsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNyQyxNQUFNLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxHQUFHLGdCQUFnQixDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUV6QyxJQUFJLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRTtZQUN0Qiw4Q0FBOEM7WUFDOUMsSUFBSSxFQUFFLEtBQUssRUFBRTtnQkFBRSxPQUFPLEVBQUUsQ0FBQztZQUN6QixNQUFNLEtBQUssR0FBRyxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUMsR0FBRyxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQztZQUNwQyxPQUFPLEVBQUUsR0FBRyxLQUFLLEdBQUcsQ0FBQyxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUM7U0FDOUI7S0FDRjtJQUVELG9FQUFvRTtJQUNwRSxPQUFPLENBQUMsQ0FBQztBQUNYLENBQUM7QUFFRCxtRkFBbUY7QUFDbkYsTUFBTSxVQUFVLEdBQUcsQ0FBQyxTQUF1QixFQUFFLE9BQW9CO0lBQy9ELElBQUksT0FBTyxDQUFDLE1BQU0sS0FBSyxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxLQUFLLENBQUMsMENBQTBDLENBQUMsQ0FBQztJQUU5RCxNQUFNLGdCQUFnQixHQUFHLFNBQVMsQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMscUJBQXFCO0lBQ3RGLE1BQU0sWUFBWSxHQUFHLEVBQUUsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGdCQUFnQixFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUUxRSxNQUFNLHFCQUFxQixHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRTtRQUNuRCxNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLHNCQUFzQixDQUFDLENBQUM7UUFDbEQsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBeUIsQ0FBQztJQUNqRCxDQUFDLENBQUMsQ0FBQztJQUVILFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRTtRQUN0QixJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7UUFDbkIsSUFBSSxRQUFRLEdBQUcsQ0FBQyxDQUFDO1FBRWpCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQ3ZDLE1BQU0sWUFBWSxHQUFHLHFCQUFxQixDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzlDLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDaEMsTUFBTSxLQUFLLEdBQUcsaUJBQWlCLENBQUMsS0FBSyxFQUFFLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUMxRCxVQUFVLElBQUksWUFBWSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUM7WUFDMUMsUUFBUSxJQUFJLFlBQVksQ0FBQyxNQUFNLENBQUM7U0FDakM7UUFFRCxPQUFPLEdBQUcsR0FBRyxDQUFDLFVBQVUsR0FBRyxRQUFRLENBQUMsQ0FBQztJQUN2QyxDQUFDLENBQUMsQ0FBQztJQUVILDhCQUE4QjtJQUM5QixTQUFTLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUNwQyxPQUFPLFlBQVksQ0FBQztBQUN0QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogZXNsaW50LWRpc2FibGUgZ3VhcmQtZm9yLWluICovXG4vKiBlc2xpbnQtZGlzYWJsZSBtYXgtbGVuICovXG5pbXBvcnQgKiBhcyBERyBmcm9tICdkYXRhZ3Jvay1hcGkvZGcnO1xuXG4vLy8gQW4gYXJyYXkgb2YgW3gsIHldIHBvaW50cyByZXByZXNlbnRpbmcgdGhlIGRlc2lyYWJpbGl0eSBsaW5lXG4vLy8gW3gsIHldIHBhaXJzIGFyZSBzb3J0ZWQgYnkgeCBpbiBhc2NlbmRpbmcgb3JkZXJcbmV4cG9ydCB0eXBlIERlc2lyYWJpbGl0eUxpbmUgPSBudW1iZXJbXVtdO1xuXG4vLy8gQSBkZXNpcmFiaWxpdHkgbGluZSB3aXRoIGl0cyB3ZWlnaHRcbmV4cG9ydCB0eXBlIFByb3BlcnR5RGVzaXJhYmlsaXR5ID0ge1xuICBsaW5lOiBEZXNpcmFiaWxpdHlMaW5lO1xuICBtaW4/OiBudW1iZXI7IC8vLyBtaW4gdmFsdWUgb2YgdGhlIHByb3BlcnR5IChvcHRpb25hbDsgdXNlZCBmb3IgZWRpdGluZyB0aGUgbGluZSlcbiAgbWF4PzogbnVtYmVyOyAvLy8gbWF4IHZhbHVlIG9mIHRoZSBwcm9wZXJ0eSAob3B0aW9uYWw7IHVzZWQgZm9yIGVkaXRpbmcgdGhlIGxpbmUpXG4gIHdlaWdodDogbnVtYmVyOyAvLy8gMC0xXG59XG5cbi8vLyBBIG1hcCBvZiBkZXNpcmFiaWxpdHkgbGluZXMgd2l0aCB0aGVpciB3ZWlnaHRzXG5leHBvcnQgdHlwZSBEZXNpcmFiaWxpdHlQcm9maWxlID0ge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIHByb3BlcnRpZXM6IHsgW2tleTogc3RyaW5nXTogUHJvcGVydHlEZXNpcmFiaWxpdHkgfTtcbn1cblxuLy8vIENhbGN1bGF0ZXMgdGhlIGRlc2lyYWJpbGl0eSBzY29yZSBmb3IgYSBnaXZlbiB4IHZhbHVlXG4vLy8gUmV0dXJucyAwIGlmIHggaXMgb3V0c2lkZSB0aGUgcmFuZ2Ugb2YgdGhlIGRlc2lyYWJpbGl0eSBsaW5lXG4vLy8gT3RoZXJ3aXNlLCByZXR1cm5zIHRoZSB5IHZhbHVlIG9mIHRoZSBkZXNpcmFiaWxpdHkgbGluZSBhdCB4XG5leHBvcnQgZnVuY3Rpb24gZGVzaXJhYmlsaXR5U2NvcmUoeDogbnVtYmVyLCBkZXNpcmFiaWxpdHlMaW5lOiBEZXNpcmFiaWxpdHlMaW5lKTogbnVtYmVyIHtcbiAgLy8gSWYgdGhlIGxpbmUgaXMgZW1wdHkgb3IgeCBpcyBvdXRzaWRlIHRoZSByYW5nZSwgcmV0dXJuIDBcbiAgaWYgKGRlc2lyYWJpbGl0eUxpbmUubGVuZ3RoID09PSAwIHx8IHggPCBkZXNpcmFiaWxpdHlMaW5lWzBdWzBdIHx8IHggPiBkZXNpcmFiaWxpdHlMaW5lW2Rlc2lyYWJpbGl0eUxpbmUubGVuZ3RoIC0gMV1bMF0pXG4gICAgcmV0dXJuIDA7XG5cbiAgLy8gRmluZCB0aGUgdHdvIHBvaW50cyB0aGF0IHggbGllcyBiZXR3ZWVuXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgZGVzaXJhYmlsaXR5TGluZS5sZW5ndGggLSAxOyBpKyspIHtcbiAgICBjb25zdCBbeDEsIHkxXSA9IGRlc2lyYWJpbGl0eUxpbmVbaV07XG4gICAgY29uc3QgW3gyLCB5Ml0gPSBkZXNpcmFiaWxpdHlMaW5lW2kgKyAxXTtcblxuICAgIGlmICh4ID49IHgxICYmIHggPD0geDIpIHtcbiAgICAgIC8vIExpbmVhciBpbnRlcnBvbGF0aW9uIGJldHdlZW4gdGhlIHR3byBwb2ludHNcbiAgICAgIGlmICh4MSA9PT0geDIpIHJldHVybiB5MTtcbiAgICAgIGNvbnN0IHNsb3BlID0gKHkyIC0geTEpIC8gKHgyIC0geDEpO1xuICAgICAgcmV0dXJuIHkxICsgc2xvcGUgKiAoeCAtIHgxKTtcbiAgICB9XG4gIH1cblxuICAvLyBTaG91bGQgbm90IGhhcHBlbiBpZiB4IGlzIHdpdGhpbiBib3VuZHMsIGJ1dCByZXR1cm4gMCBhcyBmYWxsYmFja1xuICByZXR1cm4gMDtcbn1cblxuLyoqIENhbGN1bGF0ZXMgdGhlIG11bHRpIHBhcmFtZXRlciBvcHRpbWl6YXRpb24gc2NvcmUsIDAtMTAwLCAxMDAgaXMgdGhlIG1heGltdW0gKi9cbmV4cG9ydCBmdW5jdGlvbiBtcG8oZGF0YUZyYW1lOiBERy5EYXRhRnJhbWUsIGNvbHVtbnM6IERHLkNvbHVtbltdKTogREcuQ29sdW1uIHtcbiAgaWYgKGNvbHVtbnMubGVuZ3RoID09PSAwKVxuICAgIHRocm93IG5ldyBFcnJvcignTm8gY29sdW1ucyBwcm92aWRlZCBmb3IgTVBPIGNhbGN1bGF0aW9uLicpO1xuXG4gIGNvbnN0IHJlc3VsdENvbHVtbk5hbWUgPSBkYXRhRnJhbWUuY29sdW1ucy5nZXRVbnVzZWROYW1lKCdNUE8nKTsgLy8gRW5zdXJlIHVuaXF1ZSBuYW1lXG4gIGNvbnN0IHJlc3VsdENvbHVtbiA9IERHLkNvbHVtbi5mbG9hdChyZXN1bHRDb2x1bW5OYW1lLCBjb2x1bW5zWzBdLmxlbmd0aCk7XG5cbiAgY29uc3QgZGVzaXJhYmlsaXR5VGVtcGxhdGVzID0gY29sdW1ucy5tYXAoKGNvbHVtbikgPT4ge1xuICAgIGNvbnN0IHRhZyA9IGNvbHVtbi5nZXRUYWcoJ2Rlc2lyYWJpbGl0eVRlbXBsYXRlJyk7XG4gICAgcmV0dXJuIEpTT04ucGFyc2UodGFnKSBhcyBQcm9wZXJ0eURlc2lyYWJpbGl0eTtcbiAgfSk7XG5cbiAgcmVzdWx0Q29sdW1uLmluaXQoKGkpID0+IHtcbiAgICBsZXQgdG90YWxTY29yZSA9IDA7XG4gICAgbGV0IG1heFNjb3JlID0gMDtcblxuICAgIGZvciAobGV0IGogPSAwOyBqIDwgY29sdW1ucy5sZW5ndGg7IGorKykge1xuICAgICAgY29uc3QgZGVzaXJhYmlsaXR5ID0gZGVzaXJhYmlsaXR5VGVtcGxhdGVzW2pdO1xuICAgICAgY29uc3QgdmFsdWUgPSBjb2x1bW5zW2pdLmdldChpKTtcbiAgICAgIGNvbnN0IHNjb3JlID0gZGVzaXJhYmlsaXR5U2NvcmUodmFsdWUsIGRlc2lyYWJpbGl0eS5saW5lKTtcbiAgICAgIHRvdGFsU2NvcmUgKz0gZGVzaXJhYmlsaXR5LndlaWdodCAqIHNjb3JlO1xuICAgICAgbWF4U2NvcmUgKz0gZGVzaXJhYmlsaXR5LndlaWdodDtcbiAgICB9XG5cbiAgICByZXR1cm4gMTAwICogKHRvdGFsU2NvcmUgLyBtYXhTY29yZSk7XG4gIH0pO1xuXG4gIC8vIEFkZCB0aGUgY29sdW1uIHRvIHRoZSB0YWJsZVxuICBkYXRhRnJhbWUuY29sdW1ucy5hZGQocmVzdWx0Q29sdW1uKTtcbiAgcmV0dXJuIHJlc3VsdENvbHVtbjtcbn1cbiJdfQ==
|