@vizzly/dashboard 0.14.4-dev-8252b10a001b6d371cf542597333bfa8f0f1ed5a → 0.14.4-dev-28ace990f8935dcd353f0b09527a2a1b32940e13

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.
@@ -48,12 +48,13 @@ import { useWindowWidth } from '@react-hook/window-size';
48
48
  import copy from 'copy-to-clipboard';
49
49
  import html2canvas from 'html2canvas';
50
50
  import jsPDF from 'jspdf';
51
- import { Text } from '@visx/text';
52
- import { AxisBottom, AxisLeft } from '@visx/axis';
53
- import { GridRows } from '@visx/grid';
54
51
  import { RectClipPath } from '@visx/clip-path';
55
52
  import { localPoint } from '@visx/event';
56
53
  import { bisector } from '@visx/vendor/d3-array';
54
+ import { AxisBottom as AxisBottom$1, AxisLeft as AxisLeft$1 } from '@visx/axis';
55
+ import { Text } from '@visx/text';
56
+ import { GridRows as GridRows$1 } from '@visx/grid';
57
+ import '@visx/point';
57
58
  import { WaterfallChart as WaterfallChart$4 } from './charts/src/v2/components/WaterfallChart';
58
59
  import VisibilitySensor from 'react-visibility-sensor';
59
60
  import ExcelJS from 'exceljs';
@@ -40993,72 +40994,6 @@ var Legend$1 = function Legend(_ref) {
40993
40994
  });
40994
40995
  };
40995
40996
 
40996
- var GoalLine = function GoalLine(_ref) {
40997
- var goalLine = _ref.goalLine,
40998
- innerWidth = _ref.innerWidth,
40999
- y = _ref.y,
41000
- margin = _ref.margin;
41001
- var textRef = useRef(null);
41002
- var _useState = useState(0),
41003
- labelWidth = _useState[0],
41004
- setLabelWidth = _useState[1];
41005
- useEffect(function () {
41006
- if (textRef.current) {
41007
- var bbox = textRef.current.getBBox();
41008
- setLabelWidth(bbox.width + 16);
41009
- }
41010
- }, [goalLine.formattedValue]);
41011
- var height = 16;
41012
- var triangleWidth = 8;
41013
- return jsxs("g", {
41014
- style: {
41015
- position: 'relative'
41016
- },
41017
- children: [jsx(Line, {
41018
- from: {
41019
- x: margin.left,
41020
- y: margin.top + y
41021
- },
41022
- to: {
41023
- x: margin.left + innerWidth,
41024
- y: margin.top + y
41025
- },
41026
- stroke: goalLine.color,
41027
- strokeDasharray: goalLine.strokeStyle == 'dashed' ? goalLine.strokeWidth * 2.5 + "," + goalLine.strokeWidth * 2.5 : undefined,
41028
- strokeWidth: goalLine.strokeWidth,
41029
- strokeLinecap: "round",
41030
- shapeRendering: "smooth"
41031
- }), jsxs("g", {
41032
- transform: "translate(" + (margin.left + innerWidth - labelWidth) + "," + (y + margin.top + height / 2) + ")",
41033
- children: [jsx("path", {
41034
- d: "M 0,0\n l " + -triangleWidth + "," + -height / 2 + " \n l " + triangleWidth + "," + -height / 2 + " \n h " + labelWidth + " \n a 3,3 0 0 1 3,3 \n v " + (height - 6) + " \n a 3,3 0 0 1 -3,3 \n h " + -labelWidth + " z",
41035
- fill: goalLine.color
41036
- }), jsx("text", {
41037
- ref: textRef,
41038
- x: labelWidth / 2,
41039
- y: -1,
41040
- dy: -4,
41041
- textAnchor: "middle",
41042
- fontFamily: "Arial, sans-serif",
41043
- fontSize: 9,
41044
- fill: '#fff',
41045
- children: goalLine.formattedValue
41046
- })]
41047
- })]
41048
- });
41049
- };
41050
-
41051
- /*
41052
- const strokeWidth = 2;
41053
- const height = 16;
41054
- const triangleWidth = 8;
41055
-
41056
-
41057
-
41058
-
41059
-
41060
- */
41061
-
41062
40997
  var ASSUMED_AVERAGE_CHAR_WIDTH = 8.8;
41063
40998
  function calculateWordWidth(word, avgCharWidth) {
41064
40999
  if (avgCharWidth === void 0) {
@@ -41149,25 +41084,253 @@ function useFlattenedData(xScaleKey, xScaleDataType, chart) {
41149
41084
  }, [chart.data, xScaleKey, xScaleDataType, chart.y.keys]);
41150
41085
  }
41151
41086
 
41152
- var _excluded$e = ["formattedValue"],
41153
- _excluded2$4 = ["formattedValue"];
41087
+ var GoalLine = function GoalLine(_ref) {
41088
+ var goalLine = _ref.goalLine,
41089
+ innerWidth = _ref.innerWidth,
41090
+ y = _ref.y,
41091
+ margin = _ref.margin;
41092
+ var textRef = useRef(null);
41093
+ var _useState = useState(0),
41094
+ labelWidth = _useState[0],
41095
+ setLabelWidth = _useState[1];
41096
+ useEffect(function () {
41097
+ if (textRef.current) {
41098
+ var bbox = textRef.current.getBBox();
41099
+ setLabelWidth(bbox.width + 16);
41100
+ }
41101
+ }, [goalLine.formattedValue]);
41102
+ var height = 16;
41103
+ var triangleWidth = 8;
41104
+ return jsxs("g", {
41105
+ style: {
41106
+ position: 'relative'
41107
+ },
41108
+ children: [jsx(Line, {
41109
+ from: {
41110
+ x: margin.left,
41111
+ y: margin.top + y
41112
+ },
41113
+ to: {
41114
+ x: margin.left + innerWidth,
41115
+ y: margin.top + y
41116
+ },
41117
+ stroke: goalLine.color,
41118
+ strokeDasharray: goalLine.strokeStyle == 'dashed' ? goalLine.strokeWidth * 2.5 + "," + goalLine.strokeWidth * 2.5 : undefined,
41119
+ strokeWidth: goalLine.strokeWidth,
41120
+ strokeLinecap: "round",
41121
+ shapeRendering: "smooth"
41122
+ }), jsxs("g", {
41123
+ transform: "translate(" + (margin.left + innerWidth - labelWidth) + "," + (y + margin.top + height / 2) + ")",
41124
+ children: [jsx("path", {
41125
+ d: "M 0,0\n l " + -triangleWidth + "," + -height / 2 + " \n l " + triangleWidth + "," + -height / 2 + " \n h " + labelWidth + " \n a 3,3 0 0 1 3,3 \n v " + (height - 6) + " \n a 3,3 0 0 1 -3,3 \n h " + -labelWidth + " z",
41126
+ fill: goalLine.color
41127
+ }), jsx("text", {
41128
+ ref: textRef,
41129
+ x: labelWidth / 2,
41130
+ y: -1,
41131
+ dy: -4,
41132
+ textAnchor: "middle",
41133
+ fontFamily: "Arial, sans-serif",
41134
+ fontSize: 9,
41135
+ fill: '#fff',
41136
+ children: goalLine.formattedValue
41137
+ })]
41138
+ })]
41139
+ });
41140
+ };
41141
+
41142
+ /*
41143
+ const strokeWidth = 2;
41144
+ const height = 16;
41145
+ const triangleWidth = 8;
41146
+
41147
+
41148
+
41149
+
41150
+
41151
+ */
41152
+
41153
+ var shouldUpdate$1 = function shouldUpdate(previousProps, nextProps) {
41154
+ return JSON.stringify(previousProps) == JSON.stringify(nextProps);
41155
+ };
41156
+ var GoalLines = function GoalLines(_ref) {
41157
+ var goalLines = _ref.goalLines,
41158
+ margin = _ref.margin,
41159
+ y = _ref.y,
41160
+ width = _ref.width;
41161
+ return jsx(Fragment, {
41162
+ children: goalLines.map(function (goalLine) {
41163
+ return jsx(GoalLine, {
41164
+ innerWidth: width,
41165
+ y: y(goalLine.value),
41166
+ goalLine: goalLine,
41167
+ margin: margin
41168
+ }, goalLine.value);
41169
+ })
41170
+ });
41171
+ };
41172
+ var GoalLines$1 = /*#__PURE__*/memo(GoalLines, shouldUpdate$1);
41173
+
41154
41174
  var AXIS_TITLE_STYLES = {
41155
41175
  opacity: '0.75',
41156
41176
  fontWeight: 'bold'
41157
41177
  };
41158
41178
 
41159
- /*
41160
- NOTES
41161
- -----
41162
- 1. Control width of margins via props and truncate ticks using a fixed width
41163
- 2. (somehow) prevent overlapping of ticks
41164
- 3. Split out conditional formatting colours to show operator and values for each color
41165
- 4. Add axis titles
41166
- 5. Account for no xKey and showing 1 dot for a single yKey
41167
- */
41179
+ var _excluded$e = ["formattedValue"];
41180
+ var AxisBottom = function AxisBottom(_ref) {
41181
+ var _x$ticks;
41182
+ var x = _ref.x,
41183
+ margin = _ref.margin,
41184
+ themeCSS = _ref.themeCSS,
41185
+ show = _ref.show,
41186
+ removeStroke = _ref.removeStroke,
41187
+ xScaleDataType = _ref.xScaleDataType,
41188
+ xScale = _ref.xScale,
41189
+ height = _ref.height;
41190
+ var tickFormat = useCallback(function (value) {
41191
+ var tick = null;
41192
+ if (xScaleDataType === 'date_time' && value instanceof Date) {
41193
+ var matchingTickValue = x.ticks.find(function (tickValue) {
41194
+ return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41195
+ });
41196
+ tick = matchingTickValue || null;
41197
+ } else {
41198
+ var _matchingTickValue = x.ticks.find(function (tickValue) {
41199
+ return tickValue.scaleValue === value;
41200
+ });
41201
+ tick = _matchingTickValue || null;
41202
+ }
41203
+ if (tick) {
41204
+ if (tick.formattedValue) {
41205
+ return tick.formattedValue;
41206
+ }
41207
+ return tick.value.toString();
41208
+ }
41209
+ return '';
41210
+ }, [x.ticks, xScaleDataType]);
41211
+ if (!show) return null;
41212
+ if (!xScale) return null;
41213
+ return jsx(AxisBottom$1, {
41214
+ label: x.title || undefined,
41215
+ labelProps: {
41216
+ style: AXIS_TITLE_STYLES
41217
+ },
41218
+ labelOffset: margin.bottomTitleOffset,
41219
+ hideTicks: true,
41220
+ top: height,
41221
+ scale: xScale,
41222
+ tickFormat: tickFormat,
41223
+ tickValues: ((_x$ticks = x.ticks) == null ? void 0 : _x$ticks.length) > 0 ? x.ticks.map(function (tick) {
41224
+ return (tick == null ? void 0 : tick.scaleValue) !== null ? tick == null ? void 0 : tick.scaleValue : 0;
41225
+ }) : undefined,
41226
+ tickComponent: function tickComponent(_ref2) {
41227
+ var formattedValue = _ref2.formattedValue,
41228
+ tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$e);
41229
+ return jsx(Text, _extends({
41230
+ style: themeCSS.labels
41231
+ }, tickProps, {
41232
+ children: formattedValue
41233
+ }));
41234
+ },
41235
+ stroke: themeCSS.axis.stroke,
41236
+ strokeWidth: removeStroke ? 0 : 1
41237
+ });
41238
+ };
41239
+
41240
+ var _excluded$f = ["formattedValue"];
41241
+ var AxisLeft = function AxisLeft(_ref) {
41242
+ var show = _ref.show,
41243
+ y = _ref.y,
41244
+ margin = _ref.margin,
41245
+ themeCSS = _ref.themeCSS,
41246
+ yScale = _ref.yScale,
41247
+ ticks = _ref.ticks,
41248
+ stroke = _ref.stroke;
41249
+ var tickFormat = useCallback(function (value) {
41250
+ var item = y.ticks.filter(function (tick) {
41251
+ return tick.value === value;
41252
+ })[0];
41253
+ if (item) {
41254
+ if (item.formattedValue) {
41255
+ return item.formattedValue;
41256
+ } else {
41257
+ return item.value.toString();
41258
+ }
41259
+ } else {
41260
+ return '';
41261
+ }
41262
+ }, [y.ticks]);
41263
+ if (!show) return null;
41264
+ return jsx(AxisLeft$1, {
41265
+ labelOffset: margin.leftTitleOffset,
41266
+ label: y.title || undefined,
41267
+ labelProps: {
41268
+ style: AXIS_TITLE_STYLES
41269
+ },
41270
+ hideTicks: true,
41271
+ left: 0,
41272
+ top: 0,
41273
+ scale: yScale,
41274
+ tickFormat: tickFormat,
41275
+ tickValues: ticks,
41276
+ tickComponent: function tickComponent(_ref2) {
41277
+ var formattedValue = _ref2.formattedValue,
41278
+ tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$f);
41279
+ return jsx(Text, _extends({
41280
+ width: 10,
41281
+ style: themeCSS.labels
41282
+ }, tickProps, {
41283
+ children: formattedValue
41284
+ }));
41285
+ },
41286
+ stroke: stroke != null ? stroke : 'transparent'
41287
+ });
41288
+ };
41289
+
41290
+ var GridRows = function GridRows(_ref) {
41291
+ var ticks = _ref.ticks,
41292
+ yScale = _ref.yScale,
41293
+ removeStroke = _ref.removeStroke,
41294
+ width = _ref.width,
41295
+ height = _ref.height,
41296
+ themeCSS = _ref.themeCSS;
41297
+ return jsx(GridRows$1, {
41298
+ tickValues: ticks.length > 0 ? ticks : undefined,
41299
+ scale: yScale,
41300
+ width: width,
41301
+ height: height,
41302
+ pointerEvents: "none",
41303
+ strokeDasharray: "0.5 5",
41304
+ strokeWidth: removeStroke ? 0 : 2,
41305
+ lineStyle: {
41306
+ strokeLinecap: 'round',
41307
+ stroke: themeCSS.grid.stroke
41308
+ }
41309
+ });
41310
+ };
41311
+
41312
+ function _EMOTION_STRINGIFIED_CSS_ERROR__$7() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
41313
+ var ChartWrapper$1 = function ChartWrapper(props) {
41314
+ return jsx("svg", {
41315
+ width: props.width,
41316
+ height: props.height - (props.showLegend ? 40 : 0),
41317
+ onMouseMove: props.onMouseMove,
41318
+ onMouseLeave: props.onMouseLeave,
41319
+ className: /*#__PURE__*/css$1(process.env.NODE_ENV === "production" ? {
41320
+ name: "4zleql",
41321
+ styles: "display:block"
41322
+ } : {
41323
+ name: "fx4tbw-ChartWrapper",
41324
+ styles: "display:block;label:ChartWrapper;",
41325
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkNoYXJ0V3JhcHBlci50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZ0JpQiIsImZpbGUiOiJDaGFydFdyYXBwZXIudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vY3NzJztcblxuZXhwb3J0IGNvbnN0IENoYXJ0V3JhcHBlciA9IChwcm9wczoge1xuICBjaGlsZHJlbjogUmVhY3QuUmVhY3ROb2RlO1xuICB3aWR0aDogbnVtYmVyO1xuICBoZWlnaHQ6IG51bWJlcjtcbiAgc2hvd0xlZ2VuZDogYm9vbGVhbjtcbiAgb25Nb3VzZU1vdmU/OiBSZWFjdC5Nb3VzZUV2ZW50SGFuZGxlcjxTVkdTVkdFbGVtZW50PjtcbiAgb25Nb3VzZUxlYXZlPzogUmVhY3QuTW91c2VFdmVudEhhbmRsZXI8U1ZHU1ZHRWxlbWVudD47XG59KSA9PiB7XG4gIHJldHVybiAoXG4gICAgPHN2Z1xuICAgICAgd2lkdGg9e3Byb3BzLndpZHRofVxuICAgICAgaGVpZ2h0PXtwcm9wcy5oZWlnaHQgLSAocHJvcHMuc2hvd0xlZ2VuZCA/IDQwIDogMCl9XG4gICAgICBvbk1vdXNlTW92ZT17cHJvcHMub25Nb3VzZU1vdmV9XG4gICAgICBvbk1vdXNlTGVhdmU9e3Byb3BzLm9uTW91c2VMZWF2ZX1cbiAgICAgIGNsYXNzTmFtZT17Y3NzKHsgZGlzcGxheTogJ2Jsb2NrJyB9KX1cbiAgICA+XG4gICAgICB7cHJvcHMuY2hpbGRyZW59XG4gICAgPC9zdmc+XG4gICk7XG59O1xuIl19 */",
41326
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__$7
41327
+ }),
41328
+ children: props.children
41329
+ });
41330
+ };
41168
41331
 
41169
41332
  var LineChart$5 = function LineChart(_ref) {
41170
- var _theme$axis$stroke, _theme$axis;
41333
+ var _theme$axis;
41171
41334
  var chart = _ref.chart,
41172
41335
  options = _ref.options,
41173
41336
  theme = _ref.theme,
@@ -41237,21 +41400,6 @@ var LineChart$5 = function LineChart(_ref) {
41237
41400
  });
41238
41401
  }, [innerHeight, chart.y]);
41239
41402
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41240
- var goalLines = useMemo(function () {
41241
- return chart.goalLines.map(function (goalLine) {
41242
- return jsx(GoalLine, {
41243
- innerWidth: innerWidth,
41244
- y: yScale(goalLine.value),
41245
- goalLine: goalLine,
41246
- margin: {
41247
- top: margin.top,
41248
- left: margin.left,
41249
- right: margin.right,
41250
- bottom: margin.bottom
41251
- }
41252
- }, goalLine.value);
41253
- });
41254
- }, [chart.goalLines, margin, innerWidth, yScale]);
41255
41403
  var handleMouseMove = useCallback(function (event) {
41256
41404
  if (!xKey || !xScaleKey || xScale === null) return;
41257
41405
  var tooltipData = getTooltipData({
@@ -41278,112 +41426,39 @@ var LineChart$5 = function LineChart(_ref) {
41278
41426
  return Number(tick.value);
41279
41427
  });
41280
41428
  return jsxs(React__default.Fragment, {
41281
- children: [jsxs("svg", {
41429
+ children: [jsxs(ChartWrapper$1, {
41282
41430
  width: width,
41283
- height: height - (options.showLegend ? 40 : 0),
41431
+ height: height,
41432
+ showLegend: options.showLegend,
41284
41433
  onMouseMove: handleMouseMove,
41285
41434
  onMouseLeave: handleMouseLeave,
41286
- style: {
41287
- display: 'block'
41288
- },
41289
41435
  children: [jsxs(Group$2, {
41290
41436
  left: margin.left,
41291
41437
  top: margin.top,
41292
41438
  children: [jsx(GridRows, {
41293
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41294
- scale: yScale,
41439
+ ticks: yTickValues,
41440
+ yScale: yScale,
41295
41441
  width: innerWidth,
41296
41442
  height: innerHeight,
41297
- pointerEvents: "none",
41298
- strokeDasharray: "0.5 5",
41299
- strokeWidth: options.removeStroke ? 0 : 2,
41300
- lineStyle: {
41301
- strokeLinecap: 'round',
41302
- stroke: themeCSS.grid.stroke
41303
- }
41304
- }), options.axis.showXAxisLabels && jsx(AxisBottom, {
41305
- label: chart.x.title || undefined,
41306
- labelProps: {
41307
- style: AXIS_TITLE_STYLES
41308
- },
41309
- labelOffset: margin.bottomTitleOffset,
41310
- hideTicks: true,
41311
- top: innerHeight,
41312
- scale: xScale,
41313
- tickFormat: function tickFormat(value) {
41314
- var tick = null;
41315
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41316
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41317
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41318
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41319
- });
41320
- tick = matchingTickValue || null;
41321
- } else {
41322
- var _matchingTickValue = chart.x.ticks.find(function (tickValue) {
41323
- return tickValue.scaleValue === value;
41324
- });
41325
- tick = _matchingTickValue || null;
41326
- }
41327
- if (tick) {
41328
- if (tick.formattedValue) {
41329
- return tick.formattedValue;
41330
- } else {
41331
- return tick.value.toString();
41332
- }
41333
- } else {
41334
- return ''; // ?
41335
- }
41336
- },
41337
- tickValues: chart.x.ticks.length > 0 ? chart.x.ticks.map(function (tick) {
41338
- return tick.scaleValue !== null ? tick.scaleValue : 0;
41339
- }) : undefined,
41340
- tickComponent: function tickComponent(_ref2) {
41341
- var formattedValue = _ref2.formattedValue,
41342
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$e);
41343
- return jsx(Text, _extends({
41344
- style: themeCSS.labels
41345
- }, tickProps, {
41346
- children: formattedValue
41347
- }));
41348
- },
41349
- stroke: themeCSS.axis.stroke,
41350
- strokeWidth: options.removeStroke ? 0 : 1
41351
- }), options.axis.showYAxisLabels && jsx(AxisLeft, {
41352
- labelOffset: margin.leftTitleOffset,
41353
- label: chart.y.title || undefined,
41354
- labelProps: {
41355
- style: AXIS_TITLE_STYLES
41356
- },
41357
- hideTicks: true,
41358
- left: 0,
41359
- top: 0,
41360
- scale: yScale,
41361
- tickFormat: function tickFormat(value) {
41362
- var item = chart.y.ticks.filter(function (tick) {
41363
- return tick.value === value;
41364
- })[0];
41365
- if (item) {
41366
- if (item.formattedValue) {
41367
- return item.formattedValue;
41368
- } else {
41369
- return item.value.toString();
41370
- }
41371
- } else {
41372
- return '';
41373
- }
41374
- },
41375
- tickValues: yTickValues,
41376
- tickComponent: function tickComponent(_ref3) {
41377
- var formattedValue = _ref3.formattedValue,
41378
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$4);
41379
- return jsx(Text, _extends({
41380
- width: 10,
41381
- style: themeCSS.labels
41382
- }, tickProps, {
41383
- children: formattedValue
41384
- }));
41385
- },
41386
- stroke: (_theme$axis$stroke = theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke) != null ? _theme$axis$stroke : 'transparent'
41443
+ removeStroke: options.removeStroke,
41444
+ themeCSS: themeCSS
41445
+ }), jsx(AxisBottom, {
41446
+ x: chart.x,
41447
+ margin: margin,
41448
+ themeCSS: themeCSS,
41449
+ show: options.axis.showXAxisLabels,
41450
+ removeStroke: options.removeStroke,
41451
+ xScaleDataType: xScaleDataType,
41452
+ xScale: xScale,
41453
+ height: innerHeight
41454
+ }), jsx(AxisLeft, {
41455
+ show: options.axis.showYAxisLabels,
41456
+ y: chart.y,
41457
+ margin: margin,
41458
+ themeCSS: themeCSS,
41459
+ yScale: yScale,
41460
+ ticks: yTickValues,
41461
+ stroke: theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke
41387
41462
  }), jsxs(Group$2, {
41388
41463
  children: [chart.y.keys.map(function (yKey) {
41389
41464
  var _chart$lines$filter$;
@@ -41513,7 +41588,14 @@ var LineChart$5 = function LineChart(_ref) {
41513
41588
  return null;
41514
41589
  }
41515
41590
  })]
41516
- }), goalLines]
41591
+ }), jsx(GoalLines$1, {
41592
+ goalLines: chart.goalLines,
41593
+ y: function y(d) {
41594
+ return yScale(d);
41595
+ },
41596
+ margin: margin,
41597
+ width: innerWidth
41598
+ })]
41517
41599
  }), options.showLegend && jsx(Legend$1, {
41518
41600
  legendItems: chart.lines,
41519
41601
  visibleYKeys: visibleYKeys,
@@ -41561,20 +41643,14 @@ function getBarFill(bars, conditionalFormattingRules, barKey, barValues) {
41561
41643
  return fill;
41562
41644
  }
41563
41645
 
41564
- var _excluded$f = ["formattedValue"],
41565
- _excluded2$5 = ["formattedValue"];
41566
- function _EMOTION_STRINGIFIED_CSS_ERROR__$7() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
41567
- var AXIS_TITLE_STYLES$1 = {
41568
- opacity: '0.75',
41569
- fontWeight: 'bold'
41570
- };
41646
+ function _EMOTION_STRINGIFIED_CSS_ERROR__$8() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
41571
41647
  var BAR_RADIUS = 2;
41572
41648
  var PADDING = {
41573
41649
  paddingOuter: 0.05,
41574
41650
  paddingInner: 0.3
41575
41651
  };
41576
41652
  var BarChart$5 = function BarChart(_ref) {
41577
- var _chart$x2, _theme$axis$stroke, _theme$axis;
41653
+ var _theme$axis;
41578
41654
  var chart = _ref.chart,
41579
41655
  width = _ref.width,
41580
41656
  height = _ref.height,
@@ -41647,21 +41723,6 @@ var BarChart$5 = function BarChart(_ref) {
41647
41723
  return undefined;
41648
41724
  }, [xScale, yScaleKeys, options.stacked, xScaleDataType]);
41649
41725
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41650
- var goalLines = useMemo(function () {
41651
- return chart.goalLines.map(function (goalLine) {
41652
- return jsx(GoalLine, {
41653
- innerWidth: innerWidth,
41654
- y: yScale(goalLine.value),
41655
- goalLine: goalLine,
41656
- margin: {
41657
- top: margin.top,
41658
- left: margin.left,
41659
- right: margin.right,
41660
- bottom: margin.bottom
41661
- }
41662
- }, goalLine.value);
41663
- });
41664
- }, [chart.goalLines, margin, innerWidth, yScale]);
41665
41726
  var handleMouseMove = useCallback(function (event) {
41666
41727
  if (!xKey || !xScaleKey || xScale === null) return;
41667
41728
  var tooltipData = getTooltipData({
@@ -41707,115 +41768,39 @@ var BarChart$5 = function BarChart(_ref) {
41707
41768
  return Number(tick.value);
41708
41769
  });
41709
41770
  return jsxs(Fragment, {
41710
- children: [jsxs("svg", {
41771
+ children: [jsxs(ChartWrapper$1, {
41711
41772
  width: width,
41712
- height: height - (options.showLegend ? 40 : 0),
41773
+ height: height,
41713
41774
  onMouseMove: handleMouseMove,
41714
41775
  onMouseLeave: handleMouseLeave,
41715
- style: {
41716
- display: 'block'
41717
- },
41776
+ showLegend: options.showLegend,
41718
41777
  children: [jsxs(Group$2, {
41719
41778
  left: margin.left,
41720
41779
  top: margin.top,
41721
41780
  children: [jsx(GridRows, {
41722
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41723
- scale: yScale,
41781
+ ticks: yTickValues,
41782
+ yScale: yScale,
41724
41783
  width: innerWidth,
41725
41784
  height: innerHeight,
41726
- pointerEvents: "none",
41727
- strokeDasharray: "0.5 5",
41728
- strokeWidth: options.removeStroke ? 0 : 2,
41729
- lineStyle: {
41730
- strokeLinecap: 'round',
41731
- stroke: themeCSS.grid.stroke
41732
- }
41733
- }), options.axis.showXAxisLabels && jsx(AxisBottom, {
41734
- label: chart.x.title || undefined,
41735
- labelProps: {
41736
- style: AXIS_TITLE_STYLES$1
41737
- },
41738
- labelOffset: margin.bottomTitleOffset,
41739
- hideTicks: true,
41740
- top: innerHeight
41741
- // @ts-ignore
41742
- ,
41743
- scale: xScale,
41744
- tickFormat: function tickFormat(value) {
41745
- var tick = null;
41746
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41747
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41748
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41749
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41750
- });
41751
- tick = matchingTickValue || null;
41752
- } else {
41753
- var _chart$x;
41754
- var _matchingTickValue = chart == null || (_chart$x = chart.x) == null ? void 0 : _chart$x.ticks.find(function (tickValue) {
41755
- return ((tickValue == null ? void 0 : tickValue.scaleValue) !== null ? tickValue == null ? void 0 : tickValue.scaleValue : 0) === value;
41756
- });
41757
- tick = _matchingTickValue || null;
41758
- }
41759
- if (tick) {
41760
- if (tick.formattedValue) {
41761
- return tick.formattedValue;
41762
- } else {
41763
- return tick.value.toString();
41764
- }
41765
- } else {
41766
- return ''; // ?
41767
- }
41768
- },
41769
- tickValues: (chart == null || (_chart$x2 = chart.x) == null || (_chart$x2 = _chart$x2.ticks) == null ? void 0 : _chart$x2.length) > 0 ? chart.x.ticks.map(function (tick) {
41770
- return (tick == null ? void 0 : tick.scaleValue) !== null ? tick == null ? void 0 : tick.scaleValue : 0;
41771
- }) : undefined,
41772
- tickComponent: function tickComponent(_ref2) {
41773
- var formattedValue = _ref2.formattedValue,
41774
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$f);
41775
- return jsx(Text, _extends({
41776
- style: themeCSS.labels
41777
- }, tickProps, {
41778
- children: formattedValue
41779
- }));
41780
- },
41781
- stroke: themeCSS.axis.stroke,
41782
- strokeWidth: options.removeStroke ? 0 : 1
41783
- }), options.axis.showYAxisLabels && jsx(AxisLeft, {
41784
- labelOffset: margin.leftTitleOffset,
41785
- label: chart.y.title || undefined,
41786
- labelProps: {
41787
- style: AXIS_TITLE_STYLES$1
41788
- },
41789
- hideTicks: true,
41790
- left: 0,
41791
- top: 0,
41792
- scale: yScale,
41793
- tickFormat: function tickFormat(value) {
41794
- var item = chart.y.ticks.filter(function (tick) {
41795
- return tick.value === value;
41796
- })[0];
41797
- if (item) {
41798
- if (item.formattedValue) {
41799
- return item.formattedValue;
41800
- } else {
41801
- return item.value.toString();
41802
- }
41803
- } else {
41804
- return '';
41805
- }
41806
- },
41807
- tickValues: yTickValues,
41808
- tickComponent: function tickComponent(_ref3) {
41809
- var formattedValue = _ref3.formattedValue,
41810
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$5);
41811
- return jsx(Text, _extends({
41812
- width: 10,
41813
- style: themeCSS.labels
41814
- }, tickProps, {
41815
- children: formattedValue
41816
- }));
41817
- },
41818
- stroke: (_theme$axis$stroke = theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke) != null ? _theme$axis$stroke : 'transparent'
41785
+ removeStroke: options.removeStroke,
41786
+ themeCSS: themeCSS
41787
+ }), jsx(AxisBottom, {
41788
+ x: chart.x,
41789
+ margin: margin,
41790
+ themeCSS: themeCSS,
41791
+ show: options.axis.showXAxisLabels,
41792
+ removeStroke: options.removeStroke,
41793
+ xScaleDataType: xScaleDataType,
41794
+ xScale: xScale,
41795
+ height: innerHeight
41796
+ }), jsx(AxisLeft, {
41797
+ show: options.axis.showYAxisLabels,
41798
+ y: chart.y,
41799
+ margin: margin,
41800
+ themeCSS: themeCSS,
41801
+ yScale: yScale,
41802
+ ticks: yTickValues,
41803
+ stroke: theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke
41819
41804
  }), jsxs(Group$2, {
41820
41805
  children: [!options.stacked && jsx(BarGroup$1, {
41821
41806
  data: dataFlattened,
@@ -41915,7 +41900,14 @@ var BarChart$5 = function BarChart(_ref) {
41915
41900
  pointerEvents: "none",
41916
41901
  opacity: 0.8
41917
41902
  })
41918
- }), goalLines]
41903
+ }), jsx(GoalLines$1, {
41904
+ goalLines: chart.goalLines,
41905
+ y: function y(d) {
41906
+ return yScale(d);
41907
+ },
41908
+ margin: margin,
41909
+ width: innerWidth
41910
+ })]
41919
41911
  }), options.showLegend && jsx(Legend$1, {
41920
41912
  legendItems: chart.bars,
41921
41913
  visibleYKeys: chart.bars.map(function (legendItem) {
@@ -41966,8 +41958,8 @@ var Bar = function Bar(props) {
41966
41958
  } : {
41967
41959
  name: "4nk3o1-Bar",
41968
41960
  styles: "cursor:pointer;label:Bar;",
41969
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["BarChart.tsx"],"names":[],"mappings":"AA8dqC","file":"BarChart.tsx","sourcesContent":["import { BarChartRepresentation } from 'shared-logic/src/BarChartV2/types';\nimport { ChartTheme } from '../../../types';\nimport { Group } from '@visx/group';\nimport { getChartThemeCSS, getTooltipData } from '../../utils';\nimport { AxisLeft, AxisBottom } from '@visx/axis';\nimport { GridRows } from '@visx/grid';\nimport { Fragment, useCallback, useMemo, useState } from 'react';\nimport { Tooltip } from '../Tooltip';\nimport { useTooltip } from '@visx/tooltip';\nimport { Legend } from '../Legend';\nimport { GoalLine } from '../GoalLine';\nimport { Text } from '@visx/text';\nimport { BarGroup, BarStack, Line } from '@visx/shape';\nimport { DataType } from 'shared-logic/src/Field/types';\nimport { scaleLinear, scaleBand } from '@visx/scale';\nimport { getBarFill } from './getBarFill';\nimport { buildMargin } from '../../utils/buildMargin';\nimport { ChartOnViewClick, DataItem, Tick } from 'shared-logic/src/ChartsV2/types';\nimport { lightenColor } from '../../../styles/color.functions';\nimport { css } from '@emotion/css';\nimport { BarGroupBar, BarGroup as BarGroupType } from '@visx/shape/lib/types';\nimport { useFlattenedData } from '../hooks/useFlattenedData';\n\n// Need to look at updating BarChart to use this type method of rendering - https://codesandbox.io/p/sandbox/visx-bar-chart-j9smpj?file=%2FExample.tsx\n// Currently it's using Stacked Chart which isn't technically correct\n\nconst AXIS_TITLE_STYLES: React.CSSProperties = { opacity: '0.75', fontWeight: 'bold' };\n\nexport type BarChartProps = {\n  width: number;\n  height: number;\n  theme?: ChartTheme;\n  chart: BarChartRepresentation;\n  options: {\n    stacked: boolean;\n    showRoundedTotal: boolean;\n    showLegend: boolean;\n    removeStroke: boolean;\n    axis: {\n      showXAxisLabels: boolean;\n      showYAxisLabels: boolean;\n    };\n  };\n  onClick: (params: ChartOnViewClick) => void;\n  enableHover?: boolean;\n};\n\nconst BAR_RADIUS = 2;\n\nconst PADDING = {\n  paddingOuter: 0.05,\n  paddingInner: 0.3,\n};\n\nexport const BarChart = ({ chart, width, height, options, theme, onClick, enableHover }: BarChartProps) => {\n  const { tooltipOpen, tooltipLeft = 0, tooltipTop = 0, tooltipData, hideTooltip, showTooltip } = useTooltip<{\n    [keyId: string]: DataItem;\n  } | null>();\n\n  const margin = buildMargin(chart.y.ticks, options.axis.showYAxisLabels, chart.y.title != null, chart.x.title != null);\n\n  // const [visibleYKeys, setVisibleYKeys] = useState(chart.bars.map((legendItem) => legendItem.yKey));\n\n  const innerWidth = width - margin.left - margin.right;\n  const innerHeight = height - margin.top - margin.bottom - (options.showLegend ? 40 : 0);\n  const xKey = chart.x.key;\n  const xScaleKey = chart.x.scale.key;\n  const yScaleKeys = chart.y.keys;\n  const xScaleDataType: DataType = chart.x.scale.dataType;\n\n  const xScale = useMemo(() => {\n    if (xScaleDataType === 'string') {\n      return scaleBand<string>({\n        range: [0, innerWidth],\n        domain: xScaleKey\n          ? [...chart.data.map(d => (d[xScaleKey] && d[xScaleKey].value !== null ? String(d[xScaleKey].value) : ''))]\n          : [],\n        ...PADDING,\n      });\n    }\n\n    if (xScaleDataType === 'date_time' && xScaleKey) {\n      return scaleBand<Date>({\n        range: [0, innerWidth],\n        domain: chart.data.map(d => new Date(d[xScaleKey].value as string)),\n        ...PADDING,\n      });\n    }\n\n    if (xScaleDataType === 'number' && xScaleKey) {\n      return scaleBand<number>({\n        range: [0, innerWidth],\n        domain: chart.data.map(d => (d[xScaleKey] && d[xScaleKey].value !== null ? Number(d[xScaleKey].value) : 0)),\n        ...PADDING,\n      });\n    }\n\n    return null;\n  }, [innerWidth, chart.x, chart.data]);\n\n  const yScale = useMemo(\n    () =>\n      scaleLinear<number>({\n        range: [innerHeight, 0],\n        domain:\n          chart.y.scale.ordering === 'asc'\n            ? [chart.y.scale.min as number, chart.y.scale.max as number]\n            : [chart.y.scale.max as number, chart.y.scale.min as number],\n        nice: true,\n        round: true,\n      }),\n    [innerHeight, chart.y]\n  );\n\n  const innerXScale = useMemo(() => {\n    if (!options.stacked && xScale && 'bandwidth' in xScale) {\n      return scaleBand<string>({\n        range: [0, xScale?.bandwidth()],\n        domain: yScaleKeys,\n        padding: 0.1,\n      });\n    }\n\n    return undefined;\n  }, [xScale, yScaleKeys, options.stacked, xScaleDataType]);\n\n  const dataFlattened = useFlattenedData<BarChartRepresentation>(xScaleKey, xScaleDataType, chart);\n\n  const goalLines = useMemo(\n    () =>\n      chart.goalLines.map(goalLine => (\n        <GoalLine\n          key={goalLine.value}\n          innerWidth={innerWidth}\n          y={yScale(goalLine.value)}\n          goalLine={goalLine}\n          margin={{\n            top: margin.top,\n            left: margin.left,\n            right: margin.right,\n            bottom: margin.bottom,\n          }}\n        />\n      )),\n    [chart.goalLines, margin, innerWidth, yScale]\n  );\n\n  const handleMouseMove = useCallback(\n    (event: React.MouseEvent<SVGElement>) => {\n      if (!xKey || !xScaleKey || xScale === null) return;\n\n      const tooltipData = getTooltipData({\n        data: chart.data,\n        event,\n        margin,\n        xScaleKey,\n        xScaleDataType,\n        xOrdering: chart.x.scale.ordering,\n        xScale,\n        chartType: 'bar',\n      });\n\n      showTooltip({\n        tooltipLeft: tooltipData?.tooltipLeft,\n        tooltipTop: tooltipData?.tooltipTop,\n        tooltipData: tooltipData?.tooltipData,\n      });\n    },\n    [showTooltip, xScale, margin, xKey, xScaleKey, xScaleDataType, chart.x.scale.ordering, chart.data]\n  );\n\n  const handleMouseLeave = useCallback(() => {\n    hideTooltip();\n  }, [hideTooltip]);\n\n  const handleOnBarClick = useCallback(\n    (event: React.MouseEvent<SVGRectElement, MouseEvent>, barGroup: BarGroupType<string>, bar: BarGroupBar<string>) => {\n      onClick({\n        clickedValue: {\n          index: barGroup.index,\n          y: {\n            key: bar.key,\n            value: bar.value,\n          },\n          x: {\n            ...chart.x.ticks[barGroup.index],\n            value: chart.data[barGroup.index][xKey].value ?? '',\n            key: xKey,\n          },\n        },\n        clickEvent: event,\n      });\n    },\n    [onClick, chart.data, xKey]\n  );\n\n  const themeCSS = useMemo(() => getChartThemeCSS(theme), [theme]);\n\n  const yTickValues = chart.y.ticks.map(tick => Number(tick.value));\n\n  return (\n    <Fragment>\n      {/* width/height 100% required or useParentSize() continuously scales as parent element has no specified height */}\n      <svg\n        width={width}\n        height={height - (options.showLegend ? 40 : 0)}\n        onMouseMove={handleMouseMove}\n        onMouseLeave={handleMouseLeave}\n        style={{ display: 'block' }}\n      >\n        <Group left={margin.left} top={margin.top}>\n          <GridRows\n            tickValues={yTickValues.length > 0 ? yTickValues : undefined}\n            scale={yScale}\n            width={innerWidth}\n            height={innerHeight}\n            pointerEvents=\"none\"\n            strokeDasharray=\"0.5 5\"\n            strokeWidth={options.removeStroke ? 0 : 2}\n            lineStyle={{\n              strokeLinecap: 'round',\n              stroke: themeCSS.grid.stroke,\n            }}\n          />\n          {options.axis.showXAxisLabels && (\n            <AxisBottom\n              label={chart.x.title || undefined}\n              labelProps={{ style: AXIS_TITLE_STYLES }}\n              labelOffset={margin.bottomTitleOffset}\n              hideTicks\n              top={innerHeight}\n              // @ts-ignore\n              scale={xScale}\n              tickFormat={value => {\n                let tick: Tick<Date | string | number> | null = null;\n\n                if (xScaleDataType === 'date_time' && value instanceof Date) {\n                  // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue\n                  const matchingTickValue = chart.x.ticks.find(\n                    tickValue => tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf()\n                  );\n\n                  tick = matchingTickValue || null;\n                } else {\n                  const matchingTickValue = chart?.x?.ticks.find(\n                    tickValue => (tickValue?.scaleValue !== null ? tickValue?.scaleValue : 0) === value\n                  );\n                  tick = matchingTickValue || null;\n                }\n\n                if (tick) {\n                  if (tick.formattedValue) {\n                    return tick.formattedValue;\n                  } else {\n                    return tick.value.toString();\n                  }\n                } else {\n                  return ''; // ?\n                }\n              }}\n              tickValues={\n                chart?.x?.ticks?.length > 0\n                  ? chart.x.ticks.map(tick => (tick?.scaleValue !== null ? tick?.scaleValue : 0))\n                  : undefined\n              }\n              tickComponent={({ formattedValue, ...tickProps }) => (\n                <Text style={themeCSS.labels} {...tickProps}>\n                  {formattedValue}\n                </Text>\n              )}\n              stroke={themeCSS.axis.stroke}\n              strokeWidth={options.removeStroke ? 0 : 1}\n            />\n          )}\n          {options.axis.showYAxisLabels && (\n            <AxisLeft\n              labelOffset={margin.leftTitleOffset}\n              label={chart.y.title || undefined}\n              labelProps={{ style: AXIS_TITLE_STYLES }}\n              hideTicks\n              left={0}\n              top={0}\n              scale={yScale}\n              tickFormat={value => {\n                const item = chart.y.ticks.filter(tick => tick.value === value)[0];\n\n                if (item) {\n                  if (item.formattedValue) {\n                    return item.formattedValue;\n                  } else {\n                    return item.value.toString();\n                  }\n                } else {\n                  return '';\n                }\n              }}\n              tickValues={yTickValues}\n              tickComponent={({ formattedValue, ...tickProps }) => (\n                <Text width={10} style={themeCSS.labels} {...tickProps}>\n                  {formattedValue}\n                </Text>\n              )}\n              stroke={theme?.axis?.stroke ?? 'transparent'}\n            />\n          )}\n          <Group>\n            {!options.stacked && (\n              <BarGroup\n                data={dataFlattened}\n                keys={yScaleKeys}\n                height={innerHeight}\n                x0={d => {\n                  // @ts-ignore\n                  const xValue = d[xScaleKey];\n\n                  if (xScaleDataType === 'string') return xValue;\n\n                  const xValueAdjusted =\n                    xScaleDataType === 'date_time'\n                      ? new Date(xValue)\n                      : xScaleDataType === 'number'\n                      ? Number(xValue)\n                      : String(xValue);\n                  // @ts-ignore\n                  return xValueAdjusted;\n                }}\n                // @ts-ignore\n                x0Scale={xScale}\n                // @ts-ignore\n                x1Scale={innerXScale}\n                yScale={yScale}\n                color={() => ''}\n              >\n                {barGroups => {\n                  return barGroups.map(barGroup => (\n                    <Group key={`bar-group-${barGroup.index}-${barGroup.x0}`} left={barGroup.x0}>\n                      {barGroup.bars.map(bar => {\n                        if (bar.value === null) return null;\n                        if (bar.height < 0 || bar.width < 0) return null;\n                        return (\n                          <Bar\n                            key={`bar-group-bar-${barGroup.index}-${bar.index}-${bar.value}-${bar.key}`}\n                            x={bar.x}\n                            y={bar.y}\n                            width={bar.width}\n                            height={bar.height}\n                            fill={getBarFill(chart.bars, chart.conditionalFormattingRules, bar.key, bar.value)}\n                            onClick={e => handleOnBarClick(e, barGroup, bar)}\n                            enableHover={enableHover}\n                          />\n                        );\n                      })}\n                    </Group>\n                  ));\n                }}\n              </BarGroup>\n            )}\n\n            {/* How to we get 'scaleValue' here? Will we need to start storing scaleKey as well as xKey? */}\n            {options.stacked && (\n              <BarStack\n                x={d => {\n                  // @ts-ignore\n                  const xValue = d[xScaleKey];\n                  if (xScaleDataType === 'string') return xValue;\n\n                  const xValueAdjusted =\n                    xScaleDataType === 'date_time'\n                      ? new Date(xValue)\n                      : xScaleDataType === 'number'\n                      ? Number(xValue)\n                      : String(xValue);\n                  // @ts-ignore\n                  return xValueAdjusted;\n                }}\n                // @ts-ignore\n                xScale={xScale}\n                yScale={yScale}\n                data={dataFlattened}\n                keys={chart.y.keys}\n                color={() => ''}\n                height={innerHeight}\n                width={innerWidth}\n                offset=\"none\"\n              >\n                {barStacks =>\n                  barStacks.map(barStack =>\n                    barStack.bars.map(bar => {\n                      if (bar.height < 3) return null;\n                      const BAR_PADDING = 1.5;\n                      return (\n                        <rect\n                          key={`bar-stack-${barStack.index}-${bar.index}`}\n                          x={bar.x}\n                          y={bar.y}\n                          width={bar.width}\n                          height={bar.height - BAR_PADDING}\n                          fill={getBarFill(chart.bars, chart.conditionalFormattingRules, bar.key, bar.bar.data)}\n                          rx={BAR_RADIUS}\n                        />\n                      );\n                    })\n                  )\n                }\n              </BarStack>\n            )}\n          </Group>\n        </Group>\n        {/* ----- TOOLTIP CROSSHAIR ----- */}\n        {tooltipData && (\n          <g>\n            <Line\n              from={{ x: tooltipLeft, y: margin.top }}\n              to={{ x: tooltipLeft, y: innerHeight + margin.top }}\n              stroke={'#aaa'}\n              strokeWidth={2}\n              pointerEvents=\"none\"\n              opacity={0.8}\n            />\n          </g>\n        )}\n        {/* ----- GOAL LINES ------ */}\n        {goalLines}\n      </svg>\n      {/* ----- LEGEND ----- */}\n      {options.showLegend && (\n        <Legend\n          legendItems={chart.bars}\n          visibleYKeys={chart.bars.map(legendItem => legendItem.yKey)}\n          setVisibleYKeys={() => {}}\n          keys={chart.keys}\n          conditionalFormattingRules={chart.conditionalFormattingRules}\n          marginLeft={margin.left - margin.leftTitleOffset}\n        />\n      )}\n      {/* ----- TOOLTIP ----- */}\n      {tooltipOpen && tooltipData && xKey && (\n        <Tooltip\n          tooltipData={tooltipData}\n          tooltipLeft={tooltipLeft}\n          tooltipTop={tooltipTop}\n          xKey={xKey}\n          keys={chart.keys}\n          visibleYKeys={chart.bars.map(legendItem => legendItem.yKey)}\n          yKeys={chart.y.keys}\n          legendItems={chart.bars}\n          showRoundedTotal={options.showRoundedTotal}\n          conditionalFormattingRules={chart.conditionalFormattingRules}\n          theme={themeCSS.popoverMenus}\n        />\n      )}\n    </Fragment>\n  );\n};\n\nconst Bar = (props: {\n  onClick: (e: React.MouseEvent<SVGRectElement, MouseEvent>) => void;\n  fill?: string;\n  x: number;\n  y: number;\n  width: number;\n  height: number;\n  key: string;\n  enableHover?: boolean;\n}) => {\n  const [fillColor, setFillColor] = useState(props.fill);\n  return (\n    <rect\n      key={props.key}\n      x={props.x}\n      y={props.y}\n      width={props.width}\n      height={props.height}\n      fill={fillColor}\n      rx={BAR_RADIUS}\n      onClick={props.onClick}\n      onMouseEnter={() => props.enableHover && setFillColor(lightenColor(props.fill, 0.8))}\n      onMouseLeave={() => props.enableHover && setFillColor(props.fill)}\n      className={props.enableHover ? css({ cursor: 'pointer' }) : ''}\n    />\n  );\n};\n"]} */",
41970
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__$7
41961
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["BarChart.tsx"],"names":[],"mappings":"AAuYqC","file":"BarChart.tsx","sourcesContent":["import { BarChartRepresentation } from 'shared-logic/src/BarChartV2/types';\nimport { ChartTheme } from '../../../types';\nimport { Group } from '@visx/group';\nimport { getChartThemeCSS, getTooltipData } from '../../utils';\nimport { Fragment, useCallback, useMemo, useState } from 'react';\nimport { Tooltip } from '../Tooltip';\nimport { useTooltip } from '@visx/tooltip';\nimport { Legend } from '../Legend';\nimport { BarGroup, BarStack, Line } from '@visx/shape';\nimport { DataType } from 'shared-logic/src/Field/types';\nimport { scaleLinear, scaleBand } from '@visx/scale';\nimport { getBarFill } from './getBarFill';\nimport { buildMargin } from '../../utils/buildMargin';\nimport { ChartOnViewClick, DataItem, Tick } from 'shared-logic/src/ChartsV2/types';\nimport { lightenColor } from '../../../styles/color.functions';\nimport { css } from '@emotion/css';\nimport { BarGroupBar, BarGroup as BarGroupType } from '@visx/shape/lib/types';\nimport { useFlattenedData } from '../hooks/useFlattenedData';\nimport GoalLines from '../GoalLine/GoalLines';\nimport { AxisBottom } from '../Axis/AxisBottom';\nimport { AxisLeft } from '../Axis/AxisLeft';\nimport { GridRows } from '../GridRows/GridRows';\nimport { ChartWrapper } from '../ChartWrapper/ChartWrapper';\n\n// Need to look at updating BarChart to use this type method of rendering - https://codesandbox.io/p/sandbox/visx-bar-chart-j9smpj?file=%2FExample.tsx\n// Currently it's using Stacked Chart which isn't technically correct\n\nexport type BarChartProps = {\n  width: number;\n  height: number;\n  theme?: ChartTheme;\n  chart: BarChartRepresentation;\n  options: {\n    stacked: boolean;\n    showRoundedTotal: boolean;\n    showLegend: boolean;\n    removeStroke: boolean;\n    axis: {\n      showXAxisLabels: boolean;\n      showYAxisLabels: boolean;\n    };\n  };\n  onClick: (params: ChartOnViewClick) => void;\n  enableHover?: boolean;\n};\n\nconst BAR_RADIUS = 2;\n\nconst PADDING = {\n  paddingOuter: 0.05,\n  paddingInner: 0.3,\n};\n\nexport const BarChart = ({ chart, width, height, options, theme, onClick, enableHover }: BarChartProps) => {\n  const { tooltipOpen, tooltipLeft = 0, tooltipTop = 0, tooltipData, hideTooltip, showTooltip } = useTooltip<{\n    [keyId: string]: DataItem;\n  } | null>();\n\n  const margin = buildMargin(chart.y.ticks, options.axis.showYAxisLabels, chart.y.title != null, chart.x.title != null);\n\n  // const [visibleYKeys, setVisibleYKeys] = useState(chart.bars.map((legendItem) => legendItem.yKey));\n\n  const innerWidth = width - margin.left - margin.right;\n  const innerHeight = height - margin.top - margin.bottom - (options.showLegend ? 40 : 0);\n  const xKey = chart.x.key;\n  const xScaleKey = chart.x.scale.key;\n  const yScaleKeys = chart.y.keys;\n  const xScaleDataType: DataType = chart.x.scale.dataType;\n\n  const xScale = useMemo(() => {\n    if (xScaleDataType === 'string') {\n      return scaleBand<string>({\n        range: [0, innerWidth],\n        domain: xScaleKey\n          ? [...chart.data.map(d => (d[xScaleKey] && d[xScaleKey].value !== null ? String(d[xScaleKey].value) : ''))]\n          : [],\n        ...PADDING,\n      });\n    }\n\n    if (xScaleDataType === 'date_time' && xScaleKey) {\n      return scaleBand<Date>({\n        range: [0, innerWidth],\n        domain: chart.data.map(d => new Date(d[xScaleKey].value as string)),\n        ...PADDING,\n      });\n    }\n\n    if (xScaleDataType === 'number' && xScaleKey) {\n      return scaleBand<number>({\n        range: [0, innerWidth],\n        domain: chart.data.map(d => (d[xScaleKey] && d[xScaleKey].value !== null ? Number(d[xScaleKey].value) : 0)),\n        ...PADDING,\n      });\n    }\n\n    return null;\n  }, [innerWidth, chart.x, chart.data]);\n\n  const yScale = useMemo(\n    () =>\n      scaleLinear<number>({\n        range: [innerHeight, 0],\n        domain:\n          chart.y.scale.ordering === 'asc'\n            ? [chart.y.scale.min as number, chart.y.scale.max as number]\n            : [chart.y.scale.max as number, chart.y.scale.min as number],\n        nice: true,\n        round: true,\n      }),\n    [innerHeight, chart.y]\n  );\n\n  const innerXScale = useMemo(() => {\n    if (!options.stacked && xScale && 'bandwidth' in xScale) {\n      return scaleBand<string>({\n        range: [0, xScale?.bandwidth()],\n        domain: yScaleKeys,\n        padding: 0.1,\n      });\n    }\n\n    return undefined;\n  }, [xScale, yScaleKeys, options.stacked, xScaleDataType]);\n\n  const dataFlattened = useFlattenedData<BarChartRepresentation>(xScaleKey, xScaleDataType, chart);\n\n  const handleMouseMove = useCallback(\n    (event: React.MouseEvent<SVGElement>) => {\n      if (!xKey || !xScaleKey || xScale === null) return;\n\n      const tooltipData = getTooltipData({\n        data: chart.data,\n        event,\n        margin,\n        xScaleKey,\n        xScaleDataType,\n        xOrdering: chart.x.scale.ordering,\n        xScale,\n        chartType: 'bar',\n      });\n\n      showTooltip({\n        tooltipLeft: tooltipData?.tooltipLeft,\n        tooltipTop: tooltipData?.tooltipTop,\n        tooltipData: tooltipData?.tooltipData,\n      });\n    },\n    [showTooltip, xScale, margin, xKey, xScaleKey, xScaleDataType, chart.x.scale.ordering, chart.data]\n  );\n\n  const handleMouseLeave = useCallback(() => {\n    hideTooltip();\n  }, [hideTooltip]);\n\n  const handleOnBarClick = useCallback(\n    (event: React.MouseEvent<SVGRectElement, MouseEvent>, barGroup: BarGroupType<string>, bar: BarGroupBar<string>) => {\n      onClick({\n        clickedValue: {\n          index: barGroup.index,\n          y: {\n            key: bar.key,\n            value: bar.value,\n          },\n          x: {\n            ...chart.x.ticks[barGroup.index],\n            value: chart.data[barGroup.index][xKey].value ?? '',\n            key: xKey,\n          },\n        },\n        clickEvent: event,\n      });\n    },\n    [onClick, chart.data, xKey]\n  );\n\n  const themeCSS = useMemo(() => getChartThemeCSS(theme), [theme]);\n\n  const yTickValues = chart.y.ticks.map(tick => Number(tick.value));\n\n  return (\n    <Fragment>\n      {/* width/height 100% required or useParentSize() continuously scales as parent element has no specified height */}\n      <ChartWrapper\n        width={width}\n        height={height}\n        onMouseMove={handleMouseMove}\n        onMouseLeave={handleMouseLeave}\n        showLegend={options.showLegend}\n      >\n        <Group left={margin.left} top={margin.top}>\n          <GridRows\n            ticks={yTickValues}\n            yScale={yScale}\n            width={innerWidth}\n            height={innerHeight}\n            removeStroke={options.removeStroke}\n            themeCSS={themeCSS}\n          />\n          <AxisBottom\n            x={chart.x}\n            margin={margin}\n            themeCSS={themeCSS}\n            show={options.axis.showXAxisLabels}\n            removeStroke={options.removeStroke}\n            xScaleDataType={xScaleDataType}\n            xScale={xScale}\n            height={innerHeight}\n          />\n          <AxisLeft\n            show={options.axis.showYAxisLabels}\n            y={chart.y}\n            margin={margin}\n            themeCSS={themeCSS}\n            yScale={yScale}\n            ticks={yTickValues}\n            stroke={theme?.axis?.stroke}\n          />\n          <Group>\n            {!options.stacked && (\n              <BarGroup\n                data={dataFlattened}\n                keys={yScaleKeys}\n                height={innerHeight}\n                x0={d => {\n                  // @ts-ignore\n                  const xValue = d[xScaleKey];\n\n                  if (xScaleDataType === 'string') return xValue;\n\n                  const xValueAdjusted =\n                    xScaleDataType === 'date_time'\n                      ? new Date(xValue)\n                      : xScaleDataType === 'number'\n                      ? Number(xValue)\n                      : String(xValue);\n                  // @ts-ignore\n                  return xValueAdjusted;\n                }}\n                // @ts-ignore\n                x0Scale={xScale}\n                // @ts-ignore\n                x1Scale={innerXScale}\n                yScale={yScale}\n                color={() => ''}\n              >\n                {barGroups => {\n                  return barGroups.map(barGroup => (\n                    <Group key={`bar-group-${barGroup.index}-${barGroup.x0}`} left={barGroup.x0}>\n                      {barGroup.bars.map(bar => {\n                        if (bar.value === null) return null;\n                        if (bar.height < 0 || bar.width < 0) return null;\n                        return (\n                          <Bar\n                            key={`bar-group-bar-${barGroup.index}-${bar.index}-${bar.value}-${bar.key}`}\n                            x={bar.x}\n                            y={bar.y}\n                            width={bar.width}\n                            height={bar.height}\n                            fill={getBarFill(chart.bars, chart.conditionalFormattingRules, bar.key, bar.value)}\n                            onClick={e => handleOnBarClick(e, barGroup, bar)}\n                            enableHover={enableHover}\n                          />\n                        );\n                      })}\n                    </Group>\n                  ));\n                }}\n              </BarGroup>\n            )}\n\n            {/* How to we get 'scaleValue' here? Will we need to start storing scaleKey as well as xKey? */}\n            {options.stacked && (\n              <BarStack\n                x={d => {\n                  // @ts-ignore\n                  const xValue = d[xScaleKey];\n                  if (xScaleDataType === 'string') return xValue;\n\n                  const xValueAdjusted =\n                    xScaleDataType === 'date_time'\n                      ? new Date(xValue)\n                      : xScaleDataType === 'number'\n                      ? Number(xValue)\n                      : String(xValue);\n                  // @ts-ignore\n                  return xValueAdjusted;\n                }}\n                // @ts-ignore\n                xScale={xScale}\n                yScale={yScale}\n                data={dataFlattened}\n                keys={chart.y.keys}\n                color={() => ''}\n                height={innerHeight}\n                width={innerWidth}\n                offset=\"none\"\n              >\n                {barStacks =>\n                  barStacks.map(barStack =>\n                    barStack.bars.map(bar => {\n                      if (bar.height < 3) return null;\n                      const BAR_PADDING = 1.5;\n                      return (\n                        <rect\n                          key={`bar-stack-${barStack.index}-${bar.index}`}\n                          x={bar.x}\n                          y={bar.y}\n                          width={bar.width}\n                          height={bar.height - BAR_PADDING}\n                          fill={getBarFill(chart.bars, chart.conditionalFormattingRules, bar.key, bar.bar.data)}\n                          rx={BAR_RADIUS}\n                        />\n                      );\n                    })\n                  )\n                }\n              </BarStack>\n            )}\n          </Group>\n        </Group>\n        {/* ----- TOOLTIP CROSSHAIR ----- */}\n        {tooltipData && (\n          <g>\n            <Line\n              from={{ x: tooltipLeft, y: margin.top }}\n              to={{ x: tooltipLeft, y: innerHeight + margin.top }}\n              stroke={'#aaa'}\n              strokeWidth={2}\n              pointerEvents=\"none\"\n              opacity={0.8}\n            />\n          </g>\n        )}\n        {/* ----- GOAL LINES ------ */}\n        <GoalLines goalLines={chart.goalLines} y={d => yScale(d)} margin={margin} width={innerWidth} />\n      </ChartWrapper>\n      {/* ----- LEGEND ----- */}\n      {options.showLegend && (\n        <Legend\n          legendItems={chart.bars}\n          visibleYKeys={chart.bars.map(legendItem => legendItem.yKey)}\n          setVisibleYKeys={() => {}}\n          keys={chart.keys}\n          conditionalFormattingRules={chart.conditionalFormattingRules}\n          marginLeft={margin.left - margin.leftTitleOffset}\n        />\n      )}\n      {/* ----- TOOLTIP ----- */}\n      {tooltipOpen && tooltipData && xKey && (\n        <Tooltip\n          tooltipData={tooltipData}\n          tooltipLeft={tooltipLeft}\n          tooltipTop={tooltipTop}\n          xKey={xKey}\n          keys={chart.keys}\n          visibleYKeys={chart.bars.map(legendItem => legendItem.yKey)}\n          yKeys={chart.y.keys}\n          legendItems={chart.bars}\n          showRoundedTotal={options.showRoundedTotal}\n          conditionalFormattingRules={chart.conditionalFormattingRules}\n          theme={themeCSS.popoverMenus}\n        />\n      )}\n    </Fragment>\n  );\n};\n\nconst Bar = (props: {\n  onClick: (e: React.MouseEvent<SVGRectElement, MouseEvent>) => void;\n  fill?: string;\n  x: number;\n  y: number;\n  width: number;\n  height: number;\n  key: string;\n  enableHover?: boolean;\n}) => {\n  const [fillColor, setFillColor] = useState(props.fill);\n  return (\n    <rect\n      key={props.key}\n      x={props.x}\n      y={props.y}\n      width={props.width}\n      height={props.height}\n      fill={fillColor}\n      rx={BAR_RADIUS}\n      onClick={props.onClick}\n      onMouseEnter={() => props.enableHover && setFillColor(lightenColor(props.fill, 0.8))}\n      onMouseLeave={() => props.enableHover && setFillColor(props.fill)}\n      className={props.enableHover ? css({ cursor: 'pointer' }) : ''}\n    />\n  );\n};\n"]} */",
41962
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__$8
41971
41963
  }) : ''
41972
41964
  }, props.key);
41973
41965
  };