@vizzly/dashboard 0.14.4-dev-0abc1d1609628ad0eebc796ece9793889262bc2f → 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.
@@ -57,12 +57,12 @@ var windowSize = require('@react-hook/window-size');
57
57
  var copy = _interopDefault(require('copy-to-clipboard'));
58
58
  var html2canvas = _interopDefault(require('html2canvas'));
59
59
  var jsPDF = _interopDefault(require('jspdf'));
60
- var text$3 = require('@visx/text');
61
- var axis = require('@visx/axis');
62
- var grid = require('@visx/grid');
63
60
  var clipPath = require('@visx/clip-path');
64
61
  var event = require('@visx/event');
65
62
  var d3Array = require('@visx/vendor/d3-array');
63
+ var axis = require('@visx/axis');
64
+ var text$3 = require('@visx/text');
65
+ var grid = require('@visx/grid');
66
66
  require('@visx/point');
67
67
  var WaterfallChart$4 = require('./charts/src/v2/components/WaterfallChart');
68
68
  var VisibilitySensor = _interopDefault(require('react-visibility-sensor'));
@@ -40948,72 +40948,6 @@ var Legend$1 = function Legend(_ref) {
40948
40948
  });
40949
40949
  };
40950
40950
 
40951
- var GoalLine = function GoalLine(_ref) {
40952
- var goalLine = _ref.goalLine,
40953
- innerWidth = _ref.innerWidth,
40954
- y = _ref.y,
40955
- margin = _ref.margin;
40956
- var textRef = React.useRef(null);
40957
- var _useState = React.useState(0),
40958
- labelWidth = _useState[0],
40959
- setLabelWidth = _useState[1];
40960
- React.useEffect(function () {
40961
- if (textRef.current) {
40962
- var bbox = textRef.current.getBBox();
40963
- setLabelWidth(bbox.width + 16);
40964
- }
40965
- }, [goalLine.formattedValue]);
40966
- var height = 16;
40967
- var triangleWidth = 8;
40968
- return jsxRuntime.jsxs("g", {
40969
- style: {
40970
- position: 'relative'
40971
- },
40972
- children: [jsxRuntime.jsx(shape.Line, {
40973
- from: {
40974
- x: margin.left,
40975
- y: margin.top + y
40976
- },
40977
- to: {
40978
- x: margin.left + innerWidth,
40979
- y: margin.top + y
40980
- },
40981
- stroke: goalLine.color,
40982
- strokeDasharray: goalLine.strokeStyle == 'dashed' ? goalLine.strokeWidth * 2.5 + "," + goalLine.strokeWidth * 2.5 : undefined,
40983
- strokeWidth: goalLine.strokeWidth,
40984
- strokeLinecap: "round",
40985
- shapeRendering: "smooth"
40986
- }), jsxRuntime.jsxs("g", {
40987
- transform: "translate(" + (margin.left + innerWidth - labelWidth) + "," + (y + margin.top + height / 2) + ")",
40988
- children: [jsxRuntime.jsx("path", {
40989
- 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",
40990
- fill: goalLine.color
40991
- }), jsxRuntime.jsx("text", {
40992
- ref: textRef,
40993
- x: labelWidth / 2,
40994
- y: -1,
40995
- dy: -4,
40996
- textAnchor: "middle",
40997
- fontFamily: "Arial, sans-serif",
40998
- fontSize: 9,
40999
- fill: '#fff',
41000
- children: goalLine.formattedValue
41001
- })]
41002
- })]
41003
- });
41004
- };
41005
-
41006
- /*
41007
- const strokeWidth = 2;
41008
- const height = 16;
41009
- const triangleWidth = 8;
41010
-
41011
-
41012
-
41013
-
41014
-
41015
- */
41016
-
41017
40951
  var ASSUMED_AVERAGE_CHAR_WIDTH = 8.8;
41018
40952
  function calculateWordWidth(word, avgCharWidth) {
41019
40953
  if (avgCharWidth === void 0) {
@@ -41104,25 +41038,250 @@ function useFlattenedData(xScaleKey, xScaleDataType, chart) {
41104
41038
  }, [chart.data, xScaleKey, xScaleDataType, chart.y.keys]);
41105
41039
  }
41106
41040
 
41107
- var _excluded$e = ["formattedValue"],
41108
- _excluded2$4 = ["formattedValue"];
41041
+ var GoalLine = function GoalLine(_ref) {
41042
+ var goalLine = _ref.goalLine,
41043
+ innerWidth = _ref.innerWidth,
41044
+ y = _ref.y,
41045
+ margin = _ref.margin;
41046
+ var textRef = React.useRef(null);
41047
+ var _useState = React.useState(0),
41048
+ labelWidth = _useState[0],
41049
+ setLabelWidth = _useState[1];
41050
+ React.useEffect(function () {
41051
+ if (textRef.current) {
41052
+ var bbox = textRef.current.getBBox();
41053
+ setLabelWidth(bbox.width + 16);
41054
+ }
41055
+ }, [goalLine.formattedValue]);
41056
+ var height = 16;
41057
+ var triangleWidth = 8;
41058
+ return jsxRuntime.jsxs("g", {
41059
+ style: {
41060
+ position: 'relative'
41061
+ },
41062
+ children: [jsxRuntime.jsx(shape.Line, {
41063
+ from: {
41064
+ x: margin.left,
41065
+ y: margin.top + y
41066
+ },
41067
+ to: {
41068
+ x: margin.left + innerWidth,
41069
+ y: margin.top + y
41070
+ },
41071
+ stroke: goalLine.color,
41072
+ strokeDasharray: goalLine.strokeStyle == 'dashed' ? goalLine.strokeWidth * 2.5 + "," + goalLine.strokeWidth * 2.5 : undefined,
41073
+ strokeWidth: goalLine.strokeWidth,
41074
+ strokeLinecap: "round",
41075
+ shapeRendering: "smooth"
41076
+ }), jsxRuntime.jsxs("g", {
41077
+ transform: "translate(" + (margin.left + innerWidth - labelWidth) + "," + (y + margin.top + height / 2) + ")",
41078
+ children: [jsxRuntime.jsx("path", {
41079
+ 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",
41080
+ fill: goalLine.color
41081
+ }), jsxRuntime.jsx("text", {
41082
+ ref: textRef,
41083
+ x: labelWidth / 2,
41084
+ y: -1,
41085
+ dy: -4,
41086
+ textAnchor: "middle",
41087
+ fontFamily: "Arial, sans-serif",
41088
+ fontSize: 9,
41089
+ fill: '#fff',
41090
+ children: goalLine.formattedValue
41091
+ })]
41092
+ })]
41093
+ });
41094
+ };
41095
+
41096
+ /*
41097
+ const strokeWidth = 2;
41098
+ const height = 16;
41099
+ const triangleWidth = 8;
41100
+
41101
+
41102
+
41103
+
41104
+
41105
+ */
41106
+
41107
+ var shouldUpdate$1 = function shouldUpdate(previousProps, nextProps) {
41108
+ return JSON.stringify(previousProps) == JSON.stringify(nextProps);
41109
+ };
41110
+ var GoalLines = function GoalLines(_ref) {
41111
+ var goalLines = _ref.goalLines,
41112
+ margin = _ref.margin,
41113
+ y = _ref.y,
41114
+ width = _ref.width;
41115
+ return jsxRuntime.jsx(React.Fragment, {
41116
+ children: goalLines.map(function (goalLine) {
41117
+ return jsxRuntime.jsx(GoalLine, {
41118
+ innerWidth: width,
41119
+ y: y(goalLine.value),
41120
+ goalLine: goalLine,
41121
+ margin: margin
41122
+ }, goalLine.value);
41123
+ })
41124
+ });
41125
+ };
41126
+ var GoalLines$1 = /*#__PURE__*/React.memo(GoalLines, shouldUpdate$1);
41127
+
41109
41128
  var AXIS_TITLE_STYLES = {
41110
41129
  opacity: '0.75',
41111
41130
  fontWeight: 'bold'
41112
41131
  };
41113
41132
 
41114
- /*
41115
- NOTES
41116
- -----
41117
- 1. Control width of margins via props and truncate ticks using a fixed width
41118
- 2. (somehow) prevent overlapping of ticks
41119
- 3. Split out conditional formatting colours to show operator and values for each color
41120
- 4. Add axis titles
41121
- 5. Account for no xKey and showing 1 dot for a single yKey
41122
- */
41133
+ var _excluded$e = ["formattedValue"];
41134
+ var AxisBottom = function AxisBottom(_ref) {
41135
+ var _x$ticks;
41136
+ var x = _ref.x,
41137
+ margin = _ref.margin,
41138
+ themeCSS = _ref.themeCSS,
41139
+ show = _ref.show,
41140
+ removeStroke = _ref.removeStroke,
41141
+ xScaleDataType = _ref.xScaleDataType,
41142
+ xScale = _ref.xScale,
41143
+ height = _ref.height;
41144
+ var tickFormat = React.useCallback(function (value) {
41145
+ var tick = null;
41146
+ if (xScaleDataType === 'date_time' && value instanceof Date) {
41147
+ var matchingTickValue = x.ticks.find(function (tickValue) {
41148
+ return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41149
+ });
41150
+ tick = matchingTickValue || null;
41151
+ } else {
41152
+ var _matchingTickValue = x.ticks.find(function (tickValue) {
41153
+ return tickValue.scaleValue === value;
41154
+ });
41155
+ tick = _matchingTickValue || null;
41156
+ }
41157
+ if (tick) {
41158
+ if (tick.formattedValue) {
41159
+ return tick.formattedValue;
41160
+ }
41161
+ return tick.value.toString();
41162
+ }
41163
+ return '';
41164
+ }, [x.ticks, xScaleDataType]);
41165
+ if (!show) return null;
41166
+ if (!xScale) return null;
41167
+ return jsxRuntime.jsx(axis.AxisBottom, {
41168
+ label: x.title || undefined,
41169
+ labelProps: {
41170
+ style: AXIS_TITLE_STYLES
41171
+ },
41172
+ labelOffset: margin.bottomTitleOffset,
41173
+ hideTicks: true,
41174
+ top: height,
41175
+ scale: xScale,
41176
+ tickFormat: tickFormat,
41177
+ tickValues: ((_x$ticks = x.ticks) == null ? void 0 : _x$ticks.length) > 0 ? x.ticks.map(function (tick) {
41178
+ return (tick == null ? void 0 : tick.scaleValue) !== null ? tick == null ? void 0 : tick.scaleValue : 0;
41179
+ }) : undefined,
41180
+ tickComponent: function tickComponent(_ref2) {
41181
+ var formattedValue = _ref2.formattedValue,
41182
+ tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$e);
41183
+ return jsxRuntime.jsx(text$3.Text, _extends({
41184
+ style: themeCSS.labels
41185
+ }, tickProps, {
41186
+ children: formattedValue
41187
+ }));
41188
+ },
41189
+ stroke: themeCSS.axis.stroke,
41190
+ strokeWidth: removeStroke ? 0 : 1
41191
+ });
41192
+ };
41193
+
41194
+ var _excluded$f = ["formattedValue"];
41195
+ var AxisLeft = function AxisLeft(_ref) {
41196
+ var show = _ref.show,
41197
+ y = _ref.y,
41198
+ margin = _ref.margin,
41199
+ themeCSS = _ref.themeCSS,
41200
+ yScale = _ref.yScale,
41201
+ ticks = _ref.ticks,
41202
+ stroke = _ref.stroke;
41203
+ var tickFormat = React.useCallback(function (value) {
41204
+ var item = y.ticks.filter(function (tick) {
41205
+ return tick.value === value;
41206
+ })[0];
41207
+ if (item) {
41208
+ if (item.formattedValue) {
41209
+ return item.formattedValue;
41210
+ } else {
41211
+ return item.value.toString();
41212
+ }
41213
+ } else {
41214
+ return '';
41215
+ }
41216
+ }, [y.ticks]);
41217
+ if (!show) return null;
41218
+ return jsxRuntime.jsx(axis.AxisLeft, {
41219
+ labelOffset: margin.leftTitleOffset,
41220
+ label: y.title || undefined,
41221
+ labelProps: {
41222
+ style: AXIS_TITLE_STYLES
41223
+ },
41224
+ hideTicks: true,
41225
+ left: 0,
41226
+ top: 0,
41227
+ scale: yScale,
41228
+ tickFormat: tickFormat,
41229
+ tickValues: ticks,
41230
+ tickComponent: function tickComponent(_ref2) {
41231
+ var formattedValue = _ref2.formattedValue,
41232
+ tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$f);
41233
+ return jsxRuntime.jsx(text$3.Text, _extends({
41234
+ width: 10,
41235
+ style: themeCSS.labels
41236
+ }, tickProps, {
41237
+ children: formattedValue
41238
+ }));
41239
+ },
41240
+ stroke: stroke != null ? stroke : 'transparent'
41241
+ });
41242
+ };
41243
+
41244
+ var GridRows = function GridRows(_ref) {
41245
+ var ticks = _ref.ticks,
41246
+ yScale = _ref.yScale,
41247
+ removeStroke = _ref.removeStroke,
41248
+ width = _ref.width,
41249
+ height = _ref.height,
41250
+ themeCSS = _ref.themeCSS;
41251
+ return jsxRuntime.jsx(grid.GridRows, {
41252
+ tickValues: ticks.length > 0 ? ticks : undefined,
41253
+ scale: yScale,
41254
+ width: width,
41255
+ height: height,
41256
+ pointerEvents: "none",
41257
+ strokeDasharray: "0.5 5",
41258
+ strokeWidth: removeStroke ? 0 : 2,
41259
+ lineStyle: {
41260
+ strokeLinecap: 'round',
41261
+ stroke: themeCSS.grid.stroke
41262
+ }
41263
+ });
41264
+ };
41265
+
41266
+ 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)."; }
41267
+ var ChartWrapper$1 = function ChartWrapper(props) {
41268
+ return jsxRuntime.jsx("svg", {
41269
+ width: props.width,
41270
+ height: props.height - (props.showLegend ? 40 : 0),
41271
+ onMouseMove: props.onMouseMove,
41272
+ onMouseLeave: props.onMouseLeave,
41273
+ className: /*#__PURE__*/css$1.css( {
41274
+ name: "fx4tbw-ChartWrapper",
41275
+ styles: "display:block;label:ChartWrapper;",
41276
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkNoYXJ0V3JhcHBlci50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZ0JpQiIsImZpbGUiOiJDaGFydFdyYXBwZXIudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vY3NzJztcblxuZXhwb3J0IGNvbnN0IENoYXJ0V3JhcHBlciA9IChwcm9wczoge1xuICBjaGlsZHJlbjogUmVhY3QuUmVhY3ROb2RlO1xuICB3aWR0aDogbnVtYmVyO1xuICBoZWlnaHQ6IG51bWJlcjtcbiAgc2hvd0xlZ2VuZDogYm9vbGVhbjtcbiAgb25Nb3VzZU1vdmU/OiBSZWFjdC5Nb3VzZUV2ZW50SGFuZGxlcjxTVkdTVkdFbGVtZW50PjtcbiAgb25Nb3VzZUxlYXZlPzogUmVhY3QuTW91c2VFdmVudEhhbmRsZXI8U1ZHU1ZHRWxlbWVudD47XG59KSA9PiB7XG4gIHJldHVybiAoXG4gICAgPHN2Z1xuICAgICAgd2lkdGg9e3Byb3BzLndpZHRofVxuICAgICAgaGVpZ2h0PXtwcm9wcy5oZWlnaHQgLSAocHJvcHMuc2hvd0xlZ2VuZCA/IDQwIDogMCl9XG4gICAgICBvbk1vdXNlTW92ZT17cHJvcHMub25Nb3VzZU1vdmV9XG4gICAgICBvbk1vdXNlTGVhdmU9e3Byb3BzLm9uTW91c2VMZWF2ZX1cbiAgICAgIGNsYXNzTmFtZT17Y3NzKHsgZGlzcGxheTogJ2Jsb2NrJyB9KX1cbiAgICA+XG4gICAgICB7cHJvcHMuY2hpbGRyZW59XG4gICAgPC9zdmc+XG4gICk7XG59O1xuIl19 */",
41277
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__$7
41278
+ }),
41279
+ children: props.children
41280
+ });
41281
+ };
41123
41282
 
41124
41283
  var LineChart$5 = function LineChart(_ref) {
41125
- var _theme$axis$stroke, _theme$axis;
41284
+ var _theme$axis;
41126
41285
  var chart = _ref.chart,
41127
41286
  options = _ref.options,
41128
41287
  theme = _ref.theme,
@@ -41192,21 +41351,6 @@ var LineChart$5 = function LineChart(_ref) {
41192
41351
  });
41193
41352
  }, [innerHeight, chart.y]);
41194
41353
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41195
- var goalLines = React.useMemo(function () {
41196
- return chart.goalLines.map(function (goalLine) {
41197
- return jsxRuntime.jsx(GoalLine, {
41198
- innerWidth: innerWidth,
41199
- y: yScale(goalLine.value),
41200
- goalLine: goalLine,
41201
- margin: {
41202
- top: margin.top,
41203
- left: margin.left,
41204
- right: margin.right,
41205
- bottom: margin.bottom
41206
- }
41207
- }, goalLine.value);
41208
- });
41209
- }, [chart.goalLines, margin, innerWidth, yScale]);
41210
41354
  var handleMouseMove = React.useCallback(function (event) {
41211
41355
  if (!xKey || !xScaleKey || xScale === null) return;
41212
41356
  var tooltipData = getTooltipData({
@@ -41233,112 +41377,39 @@ var LineChart$5 = function LineChart(_ref) {
41233
41377
  return Number(tick.value);
41234
41378
  });
41235
41379
  return jsxRuntime.jsxs(React__default.Fragment, {
41236
- children: [jsxRuntime.jsxs("svg", {
41380
+ children: [jsxRuntime.jsxs(ChartWrapper$1, {
41237
41381
  width: width,
41238
- height: height - (options.showLegend ? 40 : 0),
41382
+ height: height,
41383
+ showLegend: options.showLegend,
41239
41384
  onMouseMove: handleMouseMove,
41240
41385
  onMouseLeave: handleMouseLeave,
41241
- style: {
41242
- display: 'block'
41243
- },
41244
41386
  children: [jsxRuntime.jsxs(group.Group, {
41245
41387
  left: margin.left,
41246
41388
  top: margin.top,
41247
- children: [jsxRuntime.jsx(grid.GridRows, {
41248
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41249
- scale: yScale,
41389
+ children: [jsxRuntime.jsx(GridRows, {
41390
+ ticks: yTickValues,
41391
+ yScale: yScale,
41250
41392
  width: innerWidth,
41251
41393
  height: innerHeight,
41252
- pointerEvents: "none",
41253
- strokeDasharray: "0.5 5",
41254
- strokeWidth: options.removeStroke ? 0 : 2,
41255
- lineStyle: {
41256
- strokeLinecap: 'round',
41257
- stroke: themeCSS.grid.stroke
41258
- }
41259
- }), options.axis.showXAxisLabels && jsxRuntime.jsx(axis.AxisBottom, {
41260
- label: chart.x.title || undefined,
41261
- labelProps: {
41262
- style: AXIS_TITLE_STYLES
41263
- },
41264
- labelOffset: margin.bottomTitleOffset,
41265
- hideTicks: true,
41266
- top: innerHeight,
41267
- scale: xScale,
41268
- tickFormat: function tickFormat(value) {
41269
- var tick = null;
41270
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41271
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41272
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41273
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41274
- });
41275
- tick = matchingTickValue || null;
41276
- } else {
41277
- var _matchingTickValue = chart.x.ticks.find(function (tickValue) {
41278
- return tickValue.scaleValue === value;
41279
- });
41280
- tick = _matchingTickValue || null;
41281
- }
41282
- if (tick) {
41283
- if (tick.formattedValue) {
41284
- return tick.formattedValue;
41285
- } else {
41286
- return tick.value.toString();
41287
- }
41288
- } else {
41289
- return ''; // ?
41290
- }
41291
- },
41292
- tickValues: chart.x.ticks.length > 0 ? chart.x.ticks.map(function (tick) {
41293
- return tick.scaleValue !== null ? tick.scaleValue : 0;
41294
- }) : undefined,
41295
- tickComponent: function tickComponent(_ref2) {
41296
- var formattedValue = _ref2.formattedValue,
41297
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$e);
41298
- return jsxRuntime.jsx(text$3.Text, _extends({
41299
- style: themeCSS.labels
41300
- }, tickProps, {
41301
- children: formattedValue
41302
- }));
41303
- },
41304
- stroke: themeCSS.axis.stroke,
41305
- strokeWidth: options.removeStroke ? 0 : 1
41306
- }), options.axis.showYAxisLabels && jsxRuntime.jsx(axis.AxisLeft, {
41307
- labelOffset: margin.leftTitleOffset,
41308
- label: chart.y.title || undefined,
41309
- labelProps: {
41310
- style: AXIS_TITLE_STYLES
41311
- },
41312
- hideTicks: true,
41313
- left: 0,
41314
- top: 0,
41315
- scale: yScale,
41316
- tickFormat: function tickFormat(value) {
41317
- var item = chart.y.ticks.filter(function (tick) {
41318
- return tick.value === value;
41319
- })[0];
41320
- if (item) {
41321
- if (item.formattedValue) {
41322
- return item.formattedValue;
41323
- } else {
41324
- return item.value.toString();
41325
- }
41326
- } else {
41327
- return '';
41328
- }
41329
- },
41330
- tickValues: yTickValues,
41331
- tickComponent: function tickComponent(_ref3) {
41332
- var formattedValue = _ref3.formattedValue,
41333
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$4);
41334
- return jsxRuntime.jsx(text$3.Text, _extends({
41335
- width: 10,
41336
- style: themeCSS.labels
41337
- }, tickProps, {
41338
- children: formattedValue
41339
- }));
41340
- },
41341
- stroke: (_theme$axis$stroke = theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke) != null ? _theme$axis$stroke : 'transparent'
41394
+ removeStroke: options.removeStroke,
41395
+ themeCSS: themeCSS
41396
+ }), jsxRuntime.jsx(AxisBottom, {
41397
+ x: chart.x,
41398
+ margin: margin,
41399
+ themeCSS: themeCSS,
41400
+ show: options.axis.showXAxisLabels,
41401
+ removeStroke: options.removeStroke,
41402
+ xScaleDataType: xScaleDataType,
41403
+ xScale: xScale,
41404
+ height: innerHeight
41405
+ }), jsxRuntime.jsx(AxisLeft, {
41406
+ show: options.axis.showYAxisLabels,
41407
+ y: chart.y,
41408
+ margin: margin,
41409
+ themeCSS: themeCSS,
41410
+ yScale: yScale,
41411
+ ticks: yTickValues,
41412
+ stroke: theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke
41342
41413
  }), jsxRuntime.jsxs(group.Group, {
41343
41414
  children: [chart.y.keys.map(function (yKey) {
41344
41415
  var _chart$lines$filter$;
@@ -41468,7 +41539,14 @@ var LineChart$5 = function LineChart(_ref) {
41468
41539
  return null;
41469
41540
  }
41470
41541
  })]
41471
- }), goalLines]
41542
+ }), jsxRuntime.jsx(GoalLines$1, {
41543
+ goalLines: chart.goalLines,
41544
+ y: function y(d) {
41545
+ return yScale(d);
41546
+ },
41547
+ margin: margin,
41548
+ width: innerWidth
41549
+ })]
41472
41550
  }), options.showLegend && jsxRuntime.jsx(Legend$1, {
41473
41551
  legendItems: chart.lines,
41474
41552
  visibleYKeys: visibleYKeys,
@@ -41516,20 +41594,14 @@ function getBarFill(bars, conditionalFormattingRules, barKey, barValues) {
41516
41594
  return fill;
41517
41595
  }
41518
41596
 
41519
- var _excluded$f = ["formattedValue"],
41520
- _excluded2$5 = ["formattedValue"];
41521
- 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)."; }
41522
- var AXIS_TITLE_STYLES$1 = {
41523
- opacity: '0.75',
41524
- fontWeight: 'bold'
41525
- };
41597
+ 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)."; }
41526
41598
  var BAR_RADIUS = 2;
41527
41599
  var PADDING = {
41528
41600
  paddingOuter: 0.05,
41529
41601
  paddingInner: 0.3
41530
41602
  };
41531
41603
  var BarChart$5 = function BarChart(_ref) {
41532
- var _chart$x2, _theme$axis$stroke, _theme$axis;
41604
+ var _theme$axis;
41533
41605
  var chart = _ref.chart,
41534
41606
  width = _ref.width,
41535
41607
  height = _ref.height,
@@ -41602,21 +41674,6 @@ var BarChart$5 = function BarChart(_ref) {
41602
41674
  return undefined;
41603
41675
  }, [xScale, yScaleKeys, options.stacked, xScaleDataType]);
41604
41676
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41605
- var goalLines = React.useMemo(function () {
41606
- return chart.goalLines.map(function (goalLine) {
41607
- return jsxRuntime.jsx(GoalLine, {
41608
- innerWidth: innerWidth,
41609
- y: yScale(goalLine.value),
41610
- goalLine: goalLine,
41611
- margin: {
41612
- top: margin.top,
41613
- left: margin.left,
41614
- right: margin.right,
41615
- bottom: margin.bottom
41616
- }
41617
- }, goalLine.value);
41618
- });
41619
- }, [chart.goalLines, margin, innerWidth, yScale]);
41620
41677
  var handleMouseMove = React.useCallback(function (event) {
41621
41678
  if (!xKey || !xScaleKey || xScale === null) return;
41622
41679
  var tooltipData = getTooltipData({
@@ -41662,115 +41719,39 @@ var BarChart$5 = function BarChart(_ref) {
41662
41719
  return Number(tick.value);
41663
41720
  });
41664
41721
  return jsxRuntime.jsxs(React.Fragment, {
41665
- children: [jsxRuntime.jsxs("svg", {
41722
+ children: [jsxRuntime.jsxs(ChartWrapper$1, {
41666
41723
  width: width,
41667
- height: height - (options.showLegend ? 40 : 0),
41724
+ height: height,
41668
41725
  onMouseMove: handleMouseMove,
41669
41726
  onMouseLeave: handleMouseLeave,
41670
- style: {
41671
- display: 'block'
41672
- },
41727
+ showLegend: options.showLegend,
41673
41728
  children: [jsxRuntime.jsxs(group.Group, {
41674
41729
  left: margin.left,
41675
41730
  top: margin.top,
41676
- children: [jsxRuntime.jsx(grid.GridRows, {
41677
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41678
- scale: yScale,
41731
+ children: [jsxRuntime.jsx(GridRows, {
41732
+ ticks: yTickValues,
41733
+ yScale: yScale,
41679
41734
  width: innerWidth,
41680
41735
  height: innerHeight,
41681
- pointerEvents: "none",
41682
- strokeDasharray: "0.5 5",
41683
- strokeWidth: options.removeStroke ? 0 : 2,
41684
- lineStyle: {
41685
- strokeLinecap: 'round',
41686
- stroke: themeCSS.grid.stroke
41687
- }
41688
- }), options.axis.showXAxisLabels && jsxRuntime.jsx(axis.AxisBottom, {
41689
- label: chart.x.title || undefined,
41690
- labelProps: {
41691
- style: AXIS_TITLE_STYLES$1
41692
- },
41693
- labelOffset: margin.bottomTitleOffset,
41694
- hideTicks: true,
41695
- top: innerHeight
41696
- // @ts-ignore
41697
- ,
41698
- scale: xScale,
41699
- tickFormat: function tickFormat(value) {
41700
- var tick = null;
41701
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41702
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41703
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41704
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41705
- });
41706
- tick = matchingTickValue || null;
41707
- } else {
41708
- var _chart$x;
41709
- var _matchingTickValue = chart == null || (_chart$x = chart.x) == null ? void 0 : _chart$x.ticks.find(function (tickValue) {
41710
- return ((tickValue == null ? void 0 : tickValue.scaleValue) !== null ? tickValue == null ? void 0 : tickValue.scaleValue : 0) === value;
41711
- });
41712
- tick = _matchingTickValue || null;
41713
- }
41714
- if (tick) {
41715
- if (tick.formattedValue) {
41716
- return tick.formattedValue;
41717
- } else {
41718
- return tick.value.toString();
41719
- }
41720
- } else {
41721
- return ''; // ?
41722
- }
41723
- },
41724
- 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) {
41725
- return (tick == null ? void 0 : tick.scaleValue) !== null ? tick == null ? void 0 : tick.scaleValue : 0;
41726
- }) : undefined,
41727
- tickComponent: function tickComponent(_ref2) {
41728
- var formattedValue = _ref2.formattedValue,
41729
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$f);
41730
- return jsxRuntime.jsx(text$3.Text, _extends({
41731
- style: themeCSS.labels
41732
- }, tickProps, {
41733
- children: formattedValue
41734
- }));
41735
- },
41736
- stroke: themeCSS.axis.stroke,
41737
- strokeWidth: options.removeStroke ? 0 : 1
41738
- }), options.axis.showYAxisLabels && jsxRuntime.jsx(axis.AxisLeft, {
41739
- labelOffset: margin.leftTitleOffset,
41740
- label: chart.y.title || undefined,
41741
- labelProps: {
41742
- style: AXIS_TITLE_STYLES$1
41743
- },
41744
- hideTicks: true,
41745
- left: 0,
41746
- top: 0,
41747
- scale: yScale,
41748
- tickFormat: function tickFormat(value) {
41749
- var item = chart.y.ticks.filter(function (tick) {
41750
- return tick.value === value;
41751
- })[0];
41752
- if (item) {
41753
- if (item.formattedValue) {
41754
- return item.formattedValue;
41755
- } else {
41756
- return item.value.toString();
41757
- }
41758
- } else {
41759
- return '';
41760
- }
41761
- },
41762
- tickValues: yTickValues,
41763
- tickComponent: function tickComponent(_ref3) {
41764
- var formattedValue = _ref3.formattedValue,
41765
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$5);
41766
- return jsxRuntime.jsx(text$3.Text, _extends({
41767
- width: 10,
41768
- style: themeCSS.labels
41769
- }, tickProps, {
41770
- children: formattedValue
41771
- }));
41772
- },
41773
- stroke: (_theme$axis$stroke = theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke) != null ? _theme$axis$stroke : 'transparent'
41736
+ removeStroke: options.removeStroke,
41737
+ themeCSS: themeCSS
41738
+ }), jsxRuntime.jsx(AxisBottom, {
41739
+ x: chart.x,
41740
+ margin: margin,
41741
+ themeCSS: themeCSS,
41742
+ show: options.axis.showXAxisLabels,
41743
+ removeStroke: options.removeStroke,
41744
+ xScaleDataType: xScaleDataType,
41745
+ xScale: xScale,
41746
+ height: innerHeight
41747
+ }), jsxRuntime.jsx(AxisLeft, {
41748
+ show: options.axis.showYAxisLabels,
41749
+ y: chart.y,
41750
+ margin: margin,
41751
+ themeCSS: themeCSS,
41752
+ yScale: yScale,
41753
+ ticks: yTickValues,
41754
+ stroke: theme == null || (_theme$axis = theme.axis) == null ? void 0 : _theme$axis.stroke
41774
41755
  }), jsxRuntime.jsxs(group.Group, {
41775
41756
  children: [!options.stacked && jsxRuntime.jsx(shape.BarGroup, {
41776
41757
  data: dataFlattened,
@@ -41870,7 +41851,14 @@ var BarChart$5 = function BarChart(_ref) {
41870
41851
  pointerEvents: "none",
41871
41852
  opacity: 0.8
41872
41853
  })
41873
- }), goalLines]
41854
+ }), jsxRuntime.jsx(GoalLines$1, {
41855
+ goalLines: chart.goalLines,
41856
+ y: function y(d) {
41857
+ return yScale(d);
41858
+ },
41859
+ margin: margin,
41860
+ width: innerWidth
41861
+ })]
41874
41862
  }), options.showLegend && jsxRuntime.jsx(Legend$1, {
41875
41863
  legendItems: chart.bars,
41876
41864
  visibleYKeys: chart.bars.map(function (legendItem) {
@@ -41918,8 +41906,8 @@ var Bar = function Bar(props) {
41918
41906
  className: props.enableHover ? /*#__PURE__*/css$1.css( {
41919
41907
  name: "4nk3o1-Bar",
41920
41908
  styles: "cursor:pointer;label:Bar;",
41921
- 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"]} */",
41922
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__$7
41909
+ 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"]} */",
41910
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__$8
41923
41911
  }) : ''
41924
41912
  }, props.key);
41925
41913
  };