@datagrok-libraries/statistics 1.9.2 → 1.12.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.
Files changed (34) hide show
  1. package/css/styles.css +229 -0
  2. package/package.json +3 -2
  3. package/src/fit/const.d.ts +58 -0
  4. package/src/fit/const.d.ts.map +1 -0
  5. package/src/fit/const.js +63 -0
  6. package/src/fit/fit-data.d.ts.map +1 -1
  7. package/src/fit/fit-data.js +37 -5
  8. package/src/fit/new-fit-API.d.ts +6 -0
  9. package/src/fit/new-fit-API.d.ts.map +1 -1
  10. package/src/fit/new-fit-API.js +2 -8
  11. package/src/mpo/dialogs/desirability-mode-dialog.d.ts +14 -0
  12. package/src/mpo/dialogs/desirability-mode-dialog.d.ts.map +1 -0
  13. package/src/mpo/dialogs/desirability-mode-dialog.js +198 -0
  14. package/src/mpo/editors/desirability-editor-factory.d.ts +15 -0
  15. package/src/mpo/editors/desirability-editor-factory.d.ts.map +1 -0
  16. package/src/mpo/editors/desirability-editor-factory.js +11 -0
  17. package/src/mpo/editors/mpo-categorical-editor.d.ts +24 -0
  18. package/src/mpo/editors/mpo-categorical-editor.d.ts.map +1 -0
  19. package/src/mpo/editors/mpo-categorical-editor.js +113 -0
  20. package/src/mpo/editors/mpo-line-editor.d.ts +46 -0
  21. package/src/mpo/editors/mpo-line-editor.d.ts.map +1 -0
  22. package/src/mpo/editors/mpo-line-editor.js +583 -0
  23. package/src/mpo/mpo-profile-editor.d.ts +42 -6
  24. package/src/mpo/mpo-profile-editor.d.ts.map +1 -1
  25. package/src/mpo/mpo-profile-editor.js +368 -59
  26. package/src/mpo/mpo.d.ts +53 -3
  27. package/src/mpo/mpo.d.ts.map +1 -1
  28. package/src/mpo/mpo.js +92 -16
  29. package/src/mpo/utils.d.ts +6 -0
  30. package/src/mpo/utils.d.ts.map +1 -0
  31. package/src/mpo/utils.js +17 -0
  32. package/src/mpo/mpo-line-editor.d.ts +0 -10
  33. package/src/mpo/mpo-line-editor.d.ts.map +0 -1
  34. package/src/mpo/mpo-line-editor.js +0 -258
@@ -1,258 +0,0 @@
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"]}