cx 25.4.0 → 25.5.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 (138) hide show
  1. package/dist/charts.js +2 -2
  2. package/dist/data.js +17 -3
  3. package/dist/manifest.js +651 -651
  4. package/dist/ui.js +88 -61
  5. package/dist/widgets.js +13 -13
  6. package/package.json +1 -1
  7. package/src/charts/Legend.d.ts +6 -6
  8. package/src/charts/LegendEntry.js +128 -128
  9. package/src/charts/LegendEntry.scss +27 -27
  10. package/src/charts/PieChart.d.ts +92 -92
  11. package/src/charts/RangeMarker.js +6 -2
  12. package/src/charts/axis/Axis.d.ts +113 -113
  13. package/src/charts/axis/Axis.js +280 -280
  14. package/src/charts/axis/CategoryAxis.d.ts +30 -30
  15. package/src/charts/axis/CategoryAxis.js +241 -241
  16. package/src/charts/axis/NumericAxis.d.ts +46 -46
  17. package/src/charts/axis/NumericAxis.js +351 -351
  18. package/src/charts/axis/Stack.js +55 -55
  19. package/src/charts/axis/TimeAxis.d.ts +31 -28
  20. package/src/charts/axis/TimeAxis.js +611 -611
  21. package/src/charts/helpers/PointReducer.js +47 -47
  22. package/src/charts/helpers/SnapPointFinder.js +69 -69
  23. package/src/data/AugmentedViewBase.js +77 -75
  24. package/src/data/Binding.spec.js +69 -69
  25. package/src/data/ExposedRecordView.js +75 -70
  26. package/src/data/ExposedValueView.js +73 -72
  27. package/src/data/Expression.js +229 -229
  28. package/src/data/Expression.spec.js +229 -229
  29. package/src/data/Ref.d.ts +24 -24
  30. package/src/data/Ref.spec.js +79 -79
  31. package/src/data/StoreRef.spec.js +24 -24
  32. package/src/data/StringTemplate.js +92 -92
  33. package/src/data/StringTemplate.spec.js +132 -132
  34. package/src/data/StructuredDataAccessor.d.ts +7 -7
  35. package/src/data/StructuredSelector.js +132 -132
  36. package/src/data/SubscribableView.js +54 -54
  37. package/src/data/getAccessor.spec.js +11 -11
  38. package/src/data/getSelector.js +49 -49
  39. package/src/hooks/createLocalStorageRef.d.ts +3 -3
  40. package/src/hooks/createLocalStorageRef.js +20 -20
  41. package/src/index.scss +6 -6
  42. package/src/ui/Container.js +154 -183
  43. package/src/ui/Culture.d.ts +57 -57
  44. package/src/ui/Culture.js +139 -139
  45. package/src/ui/Cx.js +3 -3
  46. package/src/ui/DataProxy.js +45 -44
  47. package/src/ui/DetachedScope.js +98 -94
  48. package/src/ui/FocusManager.js +171 -171
  49. package/src/ui/Format.js +108 -108
  50. package/src/ui/HoverSync.js +147 -147
  51. package/src/ui/Instance.d.ts +1 -1
  52. package/src/ui/Instance.js +25 -16
  53. package/src/ui/IsolatedScope.js +30 -30
  54. package/src/ui/Repeater.d.ts +61 -61
  55. package/src/ui/Rescope.js +35 -31
  56. package/src/ui/Restate.js +167 -163
  57. package/src/ui/Widget.js +184 -200
  58. package/src/ui/adapter/ArrayAdapter.js +152 -142
  59. package/src/ui/adapter/TreeAdapter.js +101 -100
  60. package/src/ui/createFunctionalComponent.d.ts +1 -1
  61. package/src/ui/createFunctionalComponent.js +31 -36
  62. package/src/ui/layout/ContentPlaceholder.d.ts +19 -19
  63. package/src/ui/layout/ContentPlaceholder.js +105 -105
  64. package/src/ui/layout/ContentPlaceholder.spec.js +579 -579
  65. package/src/ui/layout/LabelsTopLayout.js +134 -134
  66. package/src/ui/layout/exploreChildren.d.ts +12 -15
  67. package/src/ui/layout/exploreChildren.js +27 -40
  68. package/src/util/Format.js +270 -270
  69. package/src/util/date/encodeDate.d.ts +1 -1
  70. package/src/util/date/encodeDate.js +8 -8
  71. package/src/util/date/encodeDateWithTimezoneOffset.d.ts +1 -1
  72. package/src/util/date/index.d.ts +11 -11
  73. package/src/util/date/index.js +11 -11
  74. package/src/util/date/parseDateInvariant.d.ts +3 -3
  75. package/src/util/date/parseDateInvariant.js +20 -20
  76. package/src/util/debounce.js +18 -18
  77. package/src/util/getSearchQueryPredicate.js +59 -59
  78. package/src/util/index.d.ts +51 -51
  79. package/src/util/index.js +54 -54
  80. package/src/util/isValidIdentifierName.d.ts +1 -1
  81. package/src/util/isValidIdentifierName.js +5 -5
  82. package/src/util/isValidIdentifierName.spec.js +33 -33
  83. package/src/util/scss/add-rules.scss +38 -38
  84. package/src/util/validatedDebounce.js +19 -19
  85. package/src/widgets/Button.js +118 -118
  86. package/src/widgets/CxCredit.scss +37 -37
  87. package/src/widgets/HighlightedSearchText.js +36 -36
  88. package/src/widgets/HighlightedSearchText.scss +18 -18
  89. package/src/widgets/List.scss +91 -91
  90. package/src/widgets/Sandbox.js +9 -8
  91. package/src/widgets/drag-drop/DropZone.js +214 -214
  92. package/src/widgets/form/Calendar.d.ts +86 -86
  93. package/src/widgets/form/Calendar.js +618 -618
  94. package/src/widgets/form/Calendar.scss +196 -196
  95. package/src/widgets/form/Checkbox.scss +127 -127
  96. package/src/widgets/form/ColorField.js +397 -397
  97. package/src/widgets/form/ColorField.scss +96 -96
  98. package/src/widgets/form/ColorPicker.scss +283 -283
  99. package/src/widgets/form/DateTimeField.js +576 -576
  100. package/src/widgets/form/DateTimePicker.js +392 -392
  101. package/src/widgets/form/LookupField.d.ts +179 -179
  102. package/src/widgets/form/LookupField.scss +219 -219
  103. package/src/widgets/form/MonthField.d.ts +99 -99
  104. package/src/widgets/form/MonthField.js +523 -523
  105. package/src/widgets/form/MonthPicker.d.ts +76 -76
  106. package/src/widgets/form/MonthPicker.js +641 -641
  107. package/src/widgets/form/MonthPicker.scss +118 -118
  108. package/src/widgets/form/NumberField.js +459 -459
  109. package/src/widgets/form/NumberField.scss +61 -61
  110. package/src/widgets/form/Radio.scss +121 -121
  111. package/src/widgets/form/Select.scss +99 -99
  112. package/src/widgets/form/Slider.scss +118 -118
  113. package/src/widgets/form/Switch.scss +140 -140
  114. package/src/widgets/form/TextArea.scss +43 -43
  115. package/src/widgets/form/TextField.js +290 -290
  116. package/src/widgets/form/TextField.scss +55 -55
  117. package/src/widgets/form/UploadButton.d.ts +34 -34
  118. package/src/widgets/form/variables.scss +353 -353
  119. package/src/widgets/grid/Grid.d.ts +442 -442
  120. package/src/widgets/grid/Grid.js +3414 -3414
  121. package/src/widgets/grid/GridRow.js +228 -228
  122. package/src/widgets/grid/TreeNode.d.ts +23 -23
  123. package/src/widgets/grid/TreeNode.scss +88 -88
  124. package/src/widgets/grid/variables.scss +133 -133
  125. package/src/widgets/nav/LinkButton.js +128 -128
  126. package/src/widgets/nav/Menu.scss +74 -74
  127. package/src/widgets/nav/Route.js +102 -106
  128. package/src/widgets/overlay/Dropdown.js +612 -612
  129. package/src/widgets/overlay/FlyweightTooltipTracker.js +39 -39
  130. package/src/widgets/overlay/Overlay.d.ts +73 -73
  131. package/src/widgets/overlay/Tooltip.js +1 -1
  132. package/src/widgets/overlay/Window.js +202 -202
  133. package/src/widgets/overlay/captureMouse.js +124 -124
  134. package/src/widgets/overlay/createHotPromiseWindowFactory.d.ts +18 -18
  135. package/src/widgets/overlay/createHotPromiseWindowFactory.js +56 -56
  136. package/src/widgets/overlay/index.d.ts +11 -11
  137. package/src/widgets/overlay/index.js +11 -11
  138. package/src/widgets/variables.scss +144 -144
@@ -1,351 +1,351 @@
1
- import { Axis } from "./Axis";
2
- import { VDOM } from "../../ui/Widget";
3
- import { Stack } from "./Stack";
4
- import { Format } from "../../util/Format";
5
- import { isNumber } from "../../util/isNumber";
6
-
7
- export class NumericAxis extends Axis {
8
- init() {
9
- if (this.deadZone) {
10
- this.lowerDeadZone = this.deadZone;
11
- this.upperDeadZone = this.deadZone;
12
- }
13
- super.init();
14
- }
15
-
16
- declareData() {
17
- super.declareData(...arguments, {
18
- min: undefined,
19
- max: undefined,
20
- normalized: undefined,
21
- inverted: undefined,
22
- labelDivisor: undefined,
23
- format: undefined,
24
- lowerDeadZone: undefined,
25
- upperDeadZone: undefined,
26
- });
27
- }
28
-
29
- initInstance(context, instance) {
30
- instance.calculator = new NumericScale();
31
- }
32
-
33
- explore(context, instance) {
34
- super.explore(context, instance);
35
- let { min, max, normalized, inverted, lowerDeadZone, upperDeadZone } = instance.data;
36
- instance.calculator.reset(
37
- min,
38
- max,
39
- this.snapToTicks,
40
- this.tickDivisions,
41
- this.minTickDistance,
42
- this.minTickStep,
43
- this.minLabelDistance,
44
- this.minLabelTickSize,
45
- normalized,
46
- inverted,
47
- lowerDeadZone,
48
- upperDeadZone,
49
- );
50
- }
51
-
52
- render(context, instance, key) {
53
- let { data } = instance;
54
-
55
- if (!data.bounds.valid()) return null;
56
-
57
- let baseFormatter = Format.parse(data.format);
58
- let formatter = data.labelDivisor != 1 ? (v) => baseFormatter(v / data.labelDivisor) : baseFormatter;
59
-
60
- return (
61
- <g key={key} className={data.classNames} style={data.style}>
62
- {this.renderTicksAndLabels(context, instance, formatter, this.minLabelDistance)}
63
- </g>
64
- );
65
- }
66
-
67
- static XY() {
68
- return {
69
- x: { type: NumericAxis },
70
- y: { type: NumericAxis, vertical: true },
71
- };
72
- }
73
- }
74
-
75
- NumericAxis.prototype.baseClass = "numericaxis";
76
- NumericAxis.prototype.tickDivisions = [
77
- [1, 2, 10, 20, 100],
78
- [1, 5, 10, 20, 100],
79
- ];
80
-
81
- NumericAxis.prototype.snapToTicks = 1;
82
- NumericAxis.prototype.normalized = false;
83
- NumericAxis.prototype.format = "n";
84
- NumericAxis.prototype.labelDivisor = 1;
85
- NumericAxis.prototype.minLabelTickSize = 0;
86
- NumericAxis.prototype.minTickStep = 0;
87
-
88
- Axis.alias("numeric", NumericAxis);
89
-
90
- class NumericScale {
91
- reset(
92
- min,
93
- max,
94
- snapToTicks,
95
- tickDivisions,
96
- minTickDistance,
97
- minTickStep,
98
- minLabelDistance,
99
- minLabelTickSize,
100
- normalized,
101
- inverted,
102
- lowerDeadZone,
103
- upperDeadZone,
104
- ) {
105
- this.min = min;
106
- this.max = max;
107
- this.snapToTicks = snapToTicks;
108
- this.tickDivisions = tickDivisions;
109
- this.minLabelDistance = minLabelDistance;
110
- this.minLabelTickSize = minLabelTickSize;
111
- this.minTickDistance = minTickDistance;
112
- this.minTickStep = minTickStep;
113
- this.tickSizes = [];
114
- this.normalized = normalized;
115
- this.inverted = inverted;
116
- delete this.minValue;
117
- delete this.maxValue;
118
- this.stacks = {};
119
- this.lowerDeadZone = lowerDeadZone || 0;
120
- this.upperDeadZone = upperDeadZone || 0;
121
- }
122
-
123
- map(v, offset = 0) {
124
- return this.origin + (v + offset - this.scale.min + this.scale.minPadding) * this.scale.factor;
125
- }
126
-
127
- decodeValue(n) {
128
- return n;
129
- }
130
-
131
- encodeValue(v) {
132
- return v;
133
- }
134
-
135
- constrainValue(v) {
136
- return Math.max(this.scale.min, Math.min(this.scale.max, v));
137
- }
138
-
139
- trackValue(v, offset = 0, constrain = false) {
140
- let value = (v - this.origin) / this.scale.factor - offset + this.scale.min - this.scale.minPadding;
141
- if (constrain) value = this.constrainValue(v);
142
- return value;
143
- }
144
-
145
- hash() {
146
- let r = {
147
- origin: this.origin,
148
- factor: this.scale.factor,
149
- min: this.scale.min,
150
- max: this.scale.max,
151
- minPadding: this.scale.minPadding,
152
- maxPadding: this.scale.maxPadding,
153
- };
154
- r.stacks = Object.keys(this.stacks)
155
- .map((s) => this.stacks[s].info?.join(","))
156
- .join(":");
157
- return r;
158
- }
159
-
160
- isSame(x) {
161
- let hash = this.hash();
162
- let same = x && !Object.keys(hash).some((k) => x[k] !== hash[k]);
163
- this.shouldUpdate = !same;
164
- return same;
165
- }
166
-
167
- measure(a, b) {
168
- this.a = a;
169
- this.b = b;
170
-
171
- if (this.minValue != null && this.min == null) this.min = this.minValue;
172
- if (this.maxValue != null && this.max == null) this.max = this.maxValue;
173
-
174
- for (let s in this.stacks) {
175
- let info = this.stacks[s].measure(this.normalized);
176
- let [min, max, invalid] = info;
177
- if (this.min == null || min < this.min) this.min = min;
178
- if (this.max == null || max > this.max) this.max = max;
179
- this.stacks[s].info = info;
180
- }
181
-
182
- if (this.min == null) this.min = 0;
183
- if (this.max == null) this.max = this.normalized ? 1 : 100;
184
-
185
- if (this.min == this.max) {
186
- if (this.min == 0) {
187
- this.min = -1;
188
- this.max = 1;
189
- } else {
190
- let delta = Math.abs(this.min) * 0.1;
191
- this.min -= delta;
192
- this.max += delta;
193
- }
194
- }
195
-
196
- this.origin = this.inverted ? this.b : this.a;
197
-
198
- this.scale = this.getScale();
199
-
200
- this.calculateTicks();
201
- }
202
-
203
- getScale(tickSizes) {
204
- let { min, max } = this;
205
- let smin = min;
206
- let smax = max;
207
-
208
- let tickSize;
209
- if (tickSizes && isNumber(this.snapToTicks) && tickSizes.length > 0) {
210
- tickSize = tickSizes[Math.min(tickSizes.length - 1, this.snapToTicks)];
211
- smin = Math.floor(smin / tickSize) * tickSize;
212
- smax = Math.ceil(smax / tickSize) * tickSize;
213
- } else {
214
- if (this.minValue === min) smin = this.minValuePadded;
215
- if (this.maxValue === max) smax = this.maxValuePadded;
216
- }
217
-
218
- let minPadding = this.minValue === min ? Math.max(0, smin - this.minValuePadded) : 0;
219
- let maxPadding = this.maxValue === max ? Math.max(0, this.maxValuePadded - smax) : 0;
220
-
221
- let sign = this.b > this.a ? 1 : -1;
222
-
223
- let factor =
224
- smin < smax
225
- ? (Math.abs(this.b - this.a) - this.lowerDeadZone - this.upperDeadZone) /
226
- (smax - smin + minPadding + maxPadding)
227
- : 0;
228
-
229
- if (factor < 0) factor = 0;
230
-
231
- if (factor > 0 && (this.lowerDeadZone > 0 || this.upperDeadZone > 0)) {
232
- while (factor * (min - smin) < this.lowerDeadZone) smin -= this.lowerDeadZone / factor;
233
-
234
- while (factor * (smax - max) < this.upperDeadZone) smax += this.upperDeadZone / factor;
235
-
236
- if (tickSize > 0 && isNumber(this.snapToTicks)) {
237
- smin = Math.floor(smin / tickSize) * tickSize;
238
- smax = Math.ceil(smax / tickSize) * tickSize;
239
- minPadding = this.minValue === min ? Math.max(0, smin - this.minValuePadded) : 0;
240
- maxPadding = this.maxValue === max ? Math.max(0, this.maxValuePadded - smax) : 0;
241
- }
242
-
243
- factor = smin < smax ? Math.abs(this.b - this.a) / (smax - smin + minPadding + maxPadding) : 0;
244
- }
245
-
246
- return {
247
- factor: sign * (this.inverted ? -factor : factor),
248
- min: smin,
249
- max: smax,
250
- minPadding,
251
- maxPadding,
252
- };
253
- }
254
-
255
- acknowledge(value, width = 0, offset = 0) {
256
- if (value == null) return;
257
-
258
- if (this.minValue == null || value < this.minValue) {
259
- this.minValue = value;
260
- this.minValuePadded = value + offset - width / 2;
261
- }
262
- if (this.maxValue == null || value > this.maxValue) {
263
- this.maxValue = value;
264
- this.maxValuePadded = value + offset + width / 2;
265
- }
266
- }
267
-
268
- getStack(name) {
269
- let s = this.stacks[name];
270
- if (!s) s = this.stacks[name] = new Stack();
271
- return s;
272
- }
273
-
274
- stacknowledge(name, ordinal, value) {
275
- return this.getStack(name).acknowledge(ordinal, value);
276
- }
277
-
278
- stack(name, ordinal, value) {
279
- let v = this.getStack(name).stack(ordinal, value);
280
- return v != null ? this.map(v) : null;
281
- }
282
-
283
- findTickSize(minPxDist) {
284
- return this.tickSizes.find((a) => a >= this.minLabelTickSize && a * Math.abs(this.scale.factor) >= minPxDist);
285
- }
286
-
287
- getTickSizes() {
288
- return this.tickSizes;
289
- }
290
-
291
- calculateTicks() {
292
- let dist = this.minLabelDistance / Math.abs(this.scale.factor);
293
- let unit = Math.pow(10, Math.floor(Math.log10(dist)));
294
-
295
- let bestLabelDistance = Infinity;
296
- let bestTicks = [];
297
- let bestScale = this.scale;
298
-
299
- for (let i = 0; i < this.tickDivisions.length; i++) {
300
- let divs = this.tickDivisions[i];
301
- let tickSizes = divs.filter((ts) => ts >= this.minTickStep).map((ts) => ts * unit);
302
- let scale = this.getScale(tickSizes);
303
- tickSizes.forEach((size, level) => {
304
- let labelDistance = size * Math.abs(scale.factor);
305
- if (labelDistance >= this.minLabelDistance && labelDistance < bestLabelDistance) {
306
- bestScale = scale;
307
- bestTicks = tickSizes;
308
- bestLabelDistance = labelDistance;
309
- }
310
- });
311
- }
312
- this.scale = bestScale;
313
- this.tickSizes = bestTicks.filter(
314
- (ts) => ts >= this.minTickStep && ts * Math.abs(bestScale.factor) >= this.minTickDistance,
315
- );
316
- if (this.tickSizes.length > 0) {
317
- let max = this.tickSizes[this.tickSizes.length - 1];
318
- this.tickSizes.push(2 * max);
319
- this.tickSizes.push(5 * max);
320
- this.tickSizes.push(10 * max);
321
- let min = this.tickSizes[0];
322
- let dist = min * Math.abs(bestScale.factor) >= this.minTickDistance;
323
- if (min / 10 >= this.minTickStep && dist / 10 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 10);
324
- else if (min / 5 >= this.minTickStep && dist / 5 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 5);
325
- else if (min / 2 >= this.minTickStep && dist / 2 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 2);
326
- }
327
- }
328
-
329
- getTicks(tickSizes) {
330
- return tickSizes.map((size) => {
331
- let start = Math.ceil((this.scale.min - this.scale.minPadding) / size);
332
- let end = Math.floor((this.scale.max + this.scale.maxPadding) / size);
333
- let result = [];
334
- for (let i = start; i <= end; i++) result.push(i * size + 0);
335
- return result;
336
- });
337
- }
338
-
339
- mapGridlines() {
340
- let size = this.tickSizes[0];
341
- let start = Math.ceil((this.scale.min - this.scale.minPadding) / size);
342
- let end = Math.floor((this.scale.max + this.scale.maxPadding) / size);
343
- let result = [];
344
- for (let i = start; i <= end; i++) result.push(this.map(i * size));
345
- return result;
346
- }
347
-
348
- book() {
349
- Console.warn("NumericAxis does not support the autoSize flag for column and bar graphs.");
350
- }
351
- }
1
+ import { Axis } from "./Axis";
2
+ import { VDOM } from "../../ui/Widget";
3
+ import { Stack } from "./Stack";
4
+ import { Format } from "../../util/Format";
5
+ import { isNumber } from "../../util/isNumber";
6
+
7
+ export class NumericAxis extends Axis {
8
+ init() {
9
+ if (this.deadZone) {
10
+ this.lowerDeadZone = this.deadZone;
11
+ this.upperDeadZone = this.deadZone;
12
+ }
13
+ super.init();
14
+ }
15
+
16
+ declareData() {
17
+ super.declareData(...arguments, {
18
+ min: undefined,
19
+ max: undefined,
20
+ normalized: undefined,
21
+ inverted: undefined,
22
+ labelDivisor: undefined,
23
+ format: undefined,
24
+ lowerDeadZone: undefined,
25
+ upperDeadZone: undefined,
26
+ });
27
+ }
28
+
29
+ initInstance(context, instance) {
30
+ instance.calculator = new NumericScale();
31
+ }
32
+
33
+ explore(context, instance) {
34
+ super.explore(context, instance);
35
+ let { min, max, normalized, inverted, lowerDeadZone, upperDeadZone } = instance.data;
36
+ instance.calculator.reset(
37
+ min,
38
+ max,
39
+ this.snapToTicks,
40
+ this.tickDivisions,
41
+ this.minTickDistance,
42
+ this.minTickStep,
43
+ this.minLabelDistance,
44
+ this.minLabelTickSize,
45
+ normalized,
46
+ inverted,
47
+ lowerDeadZone,
48
+ upperDeadZone,
49
+ );
50
+ }
51
+
52
+ render(context, instance, key) {
53
+ let { data } = instance;
54
+
55
+ if (!data.bounds.valid()) return null;
56
+
57
+ let baseFormatter = Format.parse(data.format);
58
+ let formatter = data.labelDivisor != 1 ? (v) => baseFormatter(v / data.labelDivisor) : baseFormatter;
59
+
60
+ return (
61
+ <g key={key} className={data.classNames} style={data.style}>
62
+ {this.renderTicksAndLabels(context, instance, formatter, this.minLabelDistance)}
63
+ </g>
64
+ );
65
+ }
66
+
67
+ static XY() {
68
+ return {
69
+ x: { type: NumericAxis },
70
+ y: { type: NumericAxis, vertical: true },
71
+ };
72
+ }
73
+ }
74
+
75
+ NumericAxis.prototype.baseClass = "numericaxis";
76
+ NumericAxis.prototype.tickDivisions = [
77
+ [1, 2, 10, 20, 100],
78
+ [1, 5, 10, 20, 100],
79
+ ];
80
+
81
+ NumericAxis.prototype.snapToTicks = 1;
82
+ NumericAxis.prototype.normalized = false;
83
+ NumericAxis.prototype.format = "n";
84
+ NumericAxis.prototype.labelDivisor = 1;
85
+ NumericAxis.prototype.minLabelTickSize = 0;
86
+ NumericAxis.prototype.minTickStep = 0;
87
+
88
+ Axis.alias("numeric", NumericAxis);
89
+
90
+ class NumericScale {
91
+ reset(
92
+ min,
93
+ max,
94
+ snapToTicks,
95
+ tickDivisions,
96
+ minTickDistance,
97
+ minTickStep,
98
+ minLabelDistance,
99
+ minLabelTickSize,
100
+ normalized,
101
+ inverted,
102
+ lowerDeadZone,
103
+ upperDeadZone,
104
+ ) {
105
+ this.min = min;
106
+ this.max = max;
107
+ this.snapToTicks = snapToTicks;
108
+ this.tickDivisions = tickDivisions;
109
+ this.minLabelDistance = minLabelDistance;
110
+ this.minLabelTickSize = minLabelTickSize;
111
+ this.minTickDistance = minTickDistance;
112
+ this.minTickStep = minTickStep;
113
+ this.tickSizes = [];
114
+ this.normalized = normalized;
115
+ this.inverted = inverted;
116
+ delete this.minValue;
117
+ delete this.maxValue;
118
+ this.stacks = {};
119
+ this.lowerDeadZone = lowerDeadZone || 0;
120
+ this.upperDeadZone = upperDeadZone || 0;
121
+ }
122
+
123
+ map(v, offset = 0) {
124
+ return this.origin + (v + offset - this.scale.min + this.scale.minPadding) * this.scale.factor;
125
+ }
126
+
127
+ decodeValue(n) {
128
+ return n;
129
+ }
130
+
131
+ encodeValue(v) {
132
+ return v;
133
+ }
134
+
135
+ constrainValue(v) {
136
+ return Math.max(this.scale.min, Math.min(this.scale.max, v));
137
+ }
138
+
139
+ trackValue(v, offset = 0, constrain = false) {
140
+ let value = (v - this.origin) / this.scale.factor - offset + this.scale.min - this.scale.minPadding;
141
+ if (constrain) value = this.constrainValue(v);
142
+ return value;
143
+ }
144
+
145
+ hash() {
146
+ let r = {
147
+ origin: this.origin,
148
+ factor: this.scale.factor,
149
+ min: this.scale.min,
150
+ max: this.scale.max,
151
+ minPadding: this.scale.minPadding,
152
+ maxPadding: this.scale.maxPadding,
153
+ };
154
+ r.stacks = Object.keys(this.stacks)
155
+ .map((s) => this.stacks[s].info?.join(","))
156
+ .join(":");
157
+ return r;
158
+ }
159
+
160
+ isSame(x) {
161
+ let hash = this.hash();
162
+ let same = x && !Object.keys(hash).some((k) => x[k] !== hash[k]);
163
+ this.shouldUpdate = !same;
164
+ return same;
165
+ }
166
+
167
+ measure(a, b) {
168
+ this.a = a;
169
+ this.b = b;
170
+
171
+ if (this.minValue != null && this.min == null) this.min = this.minValue;
172
+ if (this.maxValue != null && this.max == null) this.max = this.maxValue;
173
+
174
+ for (let s in this.stacks) {
175
+ let info = this.stacks[s].measure(this.normalized);
176
+ let [min, max, invalid] = info;
177
+ if (this.min == null || min < this.min) this.min = min;
178
+ if (this.max == null || max > this.max) this.max = max;
179
+ this.stacks[s].info = info;
180
+ }
181
+
182
+ if (this.min == null) this.min = 0;
183
+ if (this.max == null) this.max = this.normalized ? 1 : 100;
184
+
185
+ if (this.min == this.max) {
186
+ if (this.min == 0) {
187
+ this.min = -1;
188
+ this.max = 1;
189
+ } else {
190
+ let delta = Math.abs(this.min) * 0.1;
191
+ this.min -= delta;
192
+ this.max += delta;
193
+ }
194
+ }
195
+
196
+ this.origin = this.inverted ? this.b : this.a;
197
+
198
+ this.scale = this.getScale();
199
+
200
+ this.calculateTicks();
201
+ }
202
+
203
+ getScale(tickSizes) {
204
+ let { min, max } = this;
205
+ let smin = min;
206
+ let smax = max;
207
+
208
+ let tickSize;
209
+ if (tickSizes && isNumber(this.snapToTicks) && tickSizes.length > 0) {
210
+ tickSize = tickSizes[Math.min(tickSizes.length - 1, this.snapToTicks)];
211
+ smin = Math.floor(smin / tickSize) * tickSize;
212
+ smax = Math.ceil(smax / tickSize) * tickSize;
213
+ } else {
214
+ if (this.minValue === min) smin = this.minValuePadded;
215
+ if (this.maxValue === max) smax = this.maxValuePadded;
216
+ }
217
+
218
+ let minPadding = this.minValue === min ? Math.max(0, smin - this.minValuePadded) : 0;
219
+ let maxPadding = this.maxValue === max ? Math.max(0, this.maxValuePadded - smax) : 0;
220
+
221
+ let sign = this.b > this.a ? 1 : -1;
222
+
223
+ let factor =
224
+ smin < smax
225
+ ? (Math.abs(this.b - this.a) - this.lowerDeadZone - this.upperDeadZone) /
226
+ (smax - smin + minPadding + maxPadding)
227
+ : 0;
228
+
229
+ if (factor < 0) factor = 0;
230
+
231
+ if (factor > 0 && (this.lowerDeadZone > 0 || this.upperDeadZone > 0)) {
232
+ while (factor * (min - smin) < this.lowerDeadZone) smin -= this.lowerDeadZone / factor;
233
+
234
+ while (factor * (smax - max) < this.upperDeadZone) smax += this.upperDeadZone / factor;
235
+
236
+ if (tickSize > 0 && isNumber(this.snapToTicks)) {
237
+ smin = Math.floor(smin / tickSize) * tickSize;
238
+ smax = Math.ceil(smax / tickSize) * tickSize;
239
+ minPadding = this.minValue === min ? Math.max(0, smin - this.minValuePadded) : 0;
240
+ maxPadding = this.maxValue === max ? Math.max(0, this.maxValuePadded - smax) : 0;
241
+ }
242
+
243
+ factor = smin < smax ? Math.abs(this.b - this.a) / (smax - smin + minPadding + maxPadding) : 0;
244
+ }
245
+
246
+ return {
247
+ factor: sign * (this.inverted ? -factor : factor),
248
+ min: smin,
249
+ max: smax,
250
+ minPadding,
251
+ maxPadding,
252
+ };
253
+ }
254
+
255
+ acknowledge(value, width = 0, offset = 0) {
256
+ if (value == null) return;
257
+
258
+ if (this.minValue == null || value < this.minValue) {
259
+ this.minValue = value;
260
+ this.minValuePadded = value + offset - width / 2;
261
+ }
262
+ if (this.maxValue == null || value > this.maxValue) {
263
+ this.maxValue = value;
264
+ this.maxValuePadded = value + offset + width / 2;
265
+ }
266
+ }
267
+
268
+ getStack(name) {
269
+ let s = this.stacks[name];
270
+ if (!s) s = this.stacks[name] = new Stack();
271
+ return s;
272
+ }
273
+
274
+ stacknowledge(name, ordinal, value) {
275
+ return this.getStack(name).acknowledge(ordinal, value);
276
+ }
277
+
278
+ stack(name, ordinal, value) {
279
+ let v = this.getStack(name).stack(ordinal, value);
280
+ return v != null ? this.map(v) : null;
281
+ }
282
+
283
+ findTickSize(minPxDist) {
284
+ return this.tickSizes.find((a) => a >= this.minLabelTickSize && a * Math.abs(this.scale.factor) >= minPxDist);
285
+ }
286
+
287
+ getTickSizes() {
288
+ return this.tickSizes;
289
+ }
290
+
291
+ calculateTicks() {
292
+ let dist = this.minLabelDistance / Math.abs(this.scale.factor);
293
+ let unit = Math.pow(10, Math.floor(Math.log10(dist)));
294
+
295
+ let bestLabelDistance = Infinity;
296
+ let bestTicks = [];
297
+ let bestScale = this.scale;
298
+
299
+ for (let i = 0; i < this.tickDivisions.length; i++) {
300
+ let divs = this.tickDivisions[i];
301
+ let tickSizes = divs.filter((ts) => ts >= this.minTickStep).map((ts) => ts * unit);
302
+ let scale = this.getScale(tickSizes);
303
+ tickSizes.forEach((size, level) => {
304
+ let labelDistance = size * Math.abs(scale.factor);
305
+ if (labelDistance >= this.minLabelDistance && labelDistance < bestLabelDistance) {
306
+ bestScale = scale;
307
+ bestTicks = tickSizes;
308
+ bestLabelDistance = labelDistance;
309
+ }
310
+ });
311
+ }
312
+ this.scale = bestScale;
313
+ this.tickSizes = bestTicks.filter(
314
+ (ts) => ts >= this.minTickStep && ts * Math.abs(bestScale.factor) >= this.minTickDistance,
315
+ );
316
+ if (this.tickSizes.length > 0) {
317
+ let max = this.tickSizes[this.tickSizes.length - 1];
318
+ this.tickSizes.push(2 * max);
319
+ this.tickSizes.push(5 * max);
320
+ this.tickSizes.push(10 * max);
321
+ let min = this.tickSizes[0];
322
+ let dist = min * Math.abs(bestScale.factor) >= this.minTickDistance;
323
+ if (min / 10 >= this.minTickStep && dist / 10 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 10);
324
+ else if (min / 5 >= this.minTickStep && dist / 5 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 5);
325
+ else if (min / 2 >= this.minTickStep && dist / 2 >= this.minTickDistance) this.tickSizes.splice(0, 0, min / 2);
326
+ }
327
+ }
328
+
329
+ getTicks(tickSizes) {
330
+ return tickSizes.map((size) => {
331
+ let start = Math.ceil((this.scale.min - this.scale.minPadding) / size);
332
+ let end = Math.floor((this.scale.max + this.scale.maxPadding) / size);
333
+ let result = [];
334
+ for (let i = start; i <= end; i++) result.push(i * size + 0);
335
+ return result;
336
+ });
337
+ }
338
+
339
+ mapGridlines() {
340
+ let size = this.tickSizes[0];
341
+ let start = Math.ceil((this.scale.min - this.scale.minPadding) / size);
342
+ let end = Math.floor((this.scale.max + this.scale.maxPadding) / size);
343
+ let result = [];
344
+ for (let i = start; i <= end; i++) result.push(this.map(i * size));
345
+ return result;
346
+ }
347
+
348
+ book() {
349
+ Console.warn("NumericAxis does not support the autoSize flag for column and bar graphs.");
350
+ }
351
+ }