@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.
@@ -57,12 +57,13 @@ 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
+ require('@visx/point');
66
67
  var WaterfallChart$4 = require('./charts/src/v2/components/WaterfallChart');
67
68
  var VisibilitySensor = _interopDefault(require('react-visibility-sensor'));
68
69
  var ExcelJS = _interopDefault(require('exceljs'));
@@ -40947,72 +40948,6 @@ var Legend$1 = function Legend(_ref) {
40947
40948
  });
40948
40949
  };
40949
40950
 
40950
- var GoalLine = function GoalLine(_ref) {
40951
- var goalLine = _ref.goalLine,
40952
- innerWidth = _ref.innerWidth,
40953
- y = _ref.y,
40954
- margin = _ref.margin;
40955
- var textRef = React.useRef(null);
40956
- var _useState = React.useState(0),
40957
- labelWidth = _useState[0],
40958
- setLabelWidth = _useState[1];
40959
- React.useEffect(function () {
40960
- if (textRef.current) {
40961
- var bbox = textRef.current.getBBox();
40962
- setLabelWidth(bbox.width + 16);
40963
- }
40964
- }, [goalLine.formattedValue]);
40965
- var height = 16;
40966
- var triangleWidth = 8;
40967
- return jsxRuntime.jsxs("g", {
40968
- style: {
40969
- position: 'relative'
40970
- },
40971
- children: [jsxRuntime.jsx(shape.Line, {
40972
- from: {
40973
- x: margin.left,
40974
- y: margin.top + y
40975
- },
40976
- to: {
40977
- x: margin.left + innerWidth,
40978
- y: margin.top + y
40979
- },
40980
- stroke: goalLine.color,
40981
- strokeDasharray: goalLine.strokeStyle == 'dashed' ? goalLine.strokeWidth * 2.5 + "," + goalLine.strokeWidth * 2.5 : undefined,
40982
- strokeWidth: goalLine.strokeWidth,
40983
- strokeLinecap: "round",
40984
- shapeRendering: "smooth"
40985
- }), jsxRuntime.jsxs("g", {
40986
- transform: "translate(" + (margin.left + innerWidth - labelWidth) + "," + (y + margin.top + height / 2) + ")",
40987
- children: [jsxRuntime.jsx("path", {
40988
- 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",
40989
- fill: goalLine.color
40990
- }), jsxRuntime.jsx("text", {
40991
- ref: textRef,
40992
- x: labelWidth / 2,
40993
- y: -1,
40994
- dy: -4,
40995
- textAnchor: "middle",
40996
- fontFamily: "Arial, sans-serif",
40997
- fontSize: 9,
40998
- fill: '#fff',
40999
- children: goalLine.formattedValue
41000
- })]
41001
- })]
41002
- });
41003
- };
41004
-
41005
- /*
41006
- const strokeWidth = 2;
41007
- const height = 16;
41008
- const triangleWidth = 8;
41009
-
41010
-
41011
-
41012
-
41013
-
41014
- */
41015
-
41016
40951
  var ASSUMED_AVERAGE_CHAR_WIDTH = 8.8;
41017
40952
  function calculateWordWidth(word, avgCharWidth) {
41018
40953
  if (avgCharWidth === void 0) {
@@ -41103,25 +41038,250 @@ function useFlattenedData(xScaleKey, xScaleDataType, chart) {
41103
41038
  }, [chart.data, xScaleKey, xScaleDataType, chart.y.keys]);
41104
41039
  }
41105
41040
 
41106
- var _excluded$e = ["formattedValue"],
41107
- _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
+
41108
41128
  var AXIS_TITLE_STYLES = {
41109
41129
  opacity: '0.75',
41110
41130
  fontWeight: 'bold'
41111
41131
  };
41112
41132
 
41113
- /*
41114
- NOTES
41115
- -----
41116
- 1. Control width of margins via props and truncate ticks using a fixed width
41117
- 2. (somehow) prevent overlapping of ticks
41118
- 3. Split out conditional formatting colours to show operator and values for each color
41119
- 4. Add axis titles
41120
- 5. Account for no xKey and showing 1 dot for a single yKey
41121
- */
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
+ };
41122
41282
 
41123
41283
  var LineChart$5 = function LineChart(_ref) {
41124
- var _theme$axis$stroke, _theme$axis;
41284
+ var _theme$axis;
41125
41285
  var chart = _ref.chart,
41126
41286
  options = _ref.options,
41127
41287
  theme = _ref.theme,
@@ -41191,21 +41351,6 @@ var LineChart$5 = function LineChart(_ref) {
41191
41351
  });
41192
41352
  }, [innerHeight, chart.y]);
41193
41353
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41194
- var goalLines = React.useMemo(function () {
41195
- return chart.goalLines.map(function (goalLine) {
41196
- return jsxRuntime.jsx(GoalLine, {
41197
- innerWidth: innerWidth,
41198
- y: yScale(goalLine.value),
41199
- goalLine: goalLine,
41200
- margin: {
41201
- top: margin.top,
41202
- left: margin.left,
41203
- right: margin.right,
41204
- bottom: margin.bottom
41205
- }
41206
- }, goalLine.value);
41207
- });
41208
- }, [chart.goalLines, margin, innerWidth, yScale]);
41209
41354
  var handleMouseMove = React.useCallback(function (event) {
41210
41355
  if (!xKey || !xScaleKey || xScale === null) return;
41211
41356
  var tooltipData = getTooltipData({
@@ -41232,112 +41377,39 @@ var LineChart$5 = function LineChart(_ref) {
41232
41377
  return Number(tick.value);
41233
41378
  });
41234
41379
  return jsxRuntime.jsxs(React__default.Fragment, {
41235
- children: [jsxRuntime.jsxs("svg", {
41380
+ children: [jsxRuntime.jsxs(ChartWrapper$1, {
41236
41381
  width: width,
41237
- height: height - (options.showLegend ? 40 : 0),
41382
+ height: height,
41383
+ showLegend: options.showLegend,
41238
41384
  onMouseMove: handleMouseMove,
41239
41385
  onMouseLeave: handleMouseLeave,
41240
- style: {
41241
- display: 'block'
41242
- },
41243
41386
  children: [jsxRuntime.jsxs(group.Group, {
41244
41387
  left: margin.left,
41245
41388
  top: margin.top,
41246
- children: [jsxRuntime.jsx(grid.GridRows, {
41247
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41248
- scale: yScale,
41389
+ children: [jsxRuntime.jsx(GridRows, {
41390
+ ticks: yTickValues,
41391
+ yScale: yScale,
41249
41392
  width: innerWidth,
41250
41393
  height: innerHeight,
41251
- pointerEvents: "none",
41252
- strokeDasharray: "0.5 5",
41253
- strokeWidth: options.removeStroke ? 0 : 2,
41254
- lineStyle: {
41255
- strokeLinecap: 'round',
41256
- stroke: themeCSS.grid.stroke
41257
- }
41258
- }), options.axis.showXAxisLabels && jsxRuntime.jsx(axis.AxisBottom, {
41259
- label: chart.x.title || undefined,
41260
- labelProps: {
41261
- style: AXIS_TITLE_STYLES
41262
- },
41263
- labelOffset: margin.bottomTitleOffset,
41264
- hideTicks: true,
41265
- top: innerHeight,
41266
- scale: xScale,
41267
- tickFormat: function tickFormat(value) {
41268
- var tick = null;
41269
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41270
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41271
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41272
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41273
- });
41274
- tick = matchingTickValue || null;
41275
- } else {
41276
- var _matchingTickValue = chart.x.ticks.find(function (tickValue) {
41277
- return tickValue.scaleValue === value;
41278
- });
41279
- tick = _matchingTickValue || null;
41280
- }
41281
- if (tick) {
41282
- if (tick.formattedValue) {
41283
- return tick.formattedValue;
41284
- } else {
41285
- return tick.value.toString();
41286
- }
41287
- } else {
41288
- return ''; // ?
41289
- }
41290
- },
41291
- tickValues: chart.x.ticks.length > 0 ? chart.x.ticks.map(function (tick) {
41292
- return tick.scaleValue !== null ? tick.scaleValue : 0;
41293
- }) : undefined,
41294
- tickComponent: function tickComponent(_ref2) {
41295
- var formattedValue = _ref2.formattedValue,
41296
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$e);
41297
- return jsxRuntime.jsx(text$3.Text, _extends({
41298
- style: themeCSS.labels
41299
- }, tickProps, {
41300
- children: formattedValue
41301
- }));
41302
- },
41303
- stroke: themeCSS.axis.stroke,
41304
- strokeWidth: options.removeStroke ? 0 : 1
41305
- }), options.axis.showYAxisLabels && jsxRuntime.jsx(axis.AxisLeft, {
41306
- labelOffset: margin.leftTitleOffset,
41307
- label: chart.y.title || undefined,
41308
- labelProps: {
41309
- style: AXIS_TITLE_STYLES
41310
- },
41311
- hideTicks: true,
41312
- left: 0,
41313
- top: 0,
41314
- scale: yScale,
41315
- tickFormat: function tickFormat(value) {
41316
- var item = chart.y.ticks.filter(function (tick) {
41317
- return tick.value === value;
41318
- })[0];
41319
- if (item) {
41320
- if (item.formattedValue) {
41321
- return item.formattedValue;
41322
- } else {
41323
- return item.value.toString();
41324
- }
41325
- } else {
41326
- return '';
41327
- }
41328
- },
41329
- tickValues: yTickValues,
41330
- tickComponent: function tickComponent(_ref3) {
41331
- var formattedValue = _ref3.formattedValue,
41332
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$4);
41333
- return jsxRuntime.jsx(text$3.Text, _extends({
41334
- width: 10,
41335
- style: themeCSS.labels
41336
- }, tickProps, {
41337
- children: formattedValue
41338
- }));
41339
- },
41340
- 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
41341
41413
  }), jsxRuntime.jsxs(group.Group, {
41342
41414
  children: [chart.y.keys.map(function (yKey) {
41343
41415
  var _chart$lines$filter$;
@@ -41467,7 +41539,14 @@ var LineChart$5 = function LineChart(_ref) {
41467
41539
  return null;
41468
41540
  }
41469
41541
  })]
41470
- }), 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
+ })]
41471
41550
  }), options.showLegend && jsxRuntime.jsx(Legend$1, {
41472
41551
  legendItems: chart.lines,
41473
41552
  visibleYKeys: visibleYKeys,
@@ -41515,20 +41594,14 @@ function getBarFill(bars, conditionalFormattingRules, barKey, barValues) {
41515
41594
  return fill;
41516
41595
  }
41517
41596
 
41518
- var _excluded$f = ["formattedValue"],
41519
- _excluded2$5 = ["formattedValue"];
41520
- 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)."; }
41521
- var AXIS_TITLE_STYLES$1 = {
41522
- opacity: '0.75',
41523
- fontWeight: 'bold'
41524
- };
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)."; }
41525
41598
  var BAR_RADIUS = 2;
41526
41599
  var PADDING = {
41527
41600
  paddingOuter: 0.05,
41528
41601
  paddingInner: 0.3
41529
41602
  };
41530
41603
  var BarChart$5 = function BarChart(_ref) {
41531
- var _chart$x2, _theme$axis$stroke, _theme$axis;
41604
+ var _theme$axis;
41532
41605
  var chart = _ref.chart,
41533
41606
  width = _ref.width,
41534
41607
  height = _ref.height,
@@ -41601,21 +41674,6 @@ var BarChart$5 = function BarChart(_ref) {
41601
41674
  return undefined;
41602
41675
  }, [xScale, yScaleKeys, options.stacked, xScaleDataType]);
41603
41676
  var dataFlattened = useFlattenedData(xScaleKey, xScaleDataType, chart);
41604
- var goalLines = React.useMemo(function () {
41605
- return chart.goalLines.map(function (goalLine) {
41606
- return jsxRuntime.jsx(GoalLine, {
41607
- innerWidth: innerWidth,
41608
- y: yScale(goalLine.value),
41609
- goalLine: goalLine,
41610
- margin: {
41611
- top: margin.top,
41612
- left: margin.left,
41613
- right: margin.right,
41614
- bottom: margin.bottom
41615
- }
41616
- }, goalLine.value);
41617
- });
41618
- }, [chart.goalLines, margin, innerWidth, yScale]);
41619
41677
  var handleMouseMove = React.useCallback(function (event) {
41620
41678
  if (!xKey || !xScaleKey || xScale === null) return;
41621
41679
  var tooltipData = getTooltipData({
@@ -41661,115 +41719,39 @@ var BarChart$5 = function BarChart(_ref) {
41661
41719
  return Number(tick.value);
41662
41720
  });
41663
41721
  return jsxRuntime.jsxs(React.Fragment, {
41664
- children: [jsxRuntime.jsxs("svg", {
41722
+ children: [jsxRuntime.jsxs(ChartWrapper$1, {
41665
41723
  width: width,
41666
- height: height - (options.showLegend ? 40 : 0),
41724
+ height: height,
41667
41725
  onMouseMove: handleMouseMove,
41668
41726
  onMouseLeave: handleMouseLeave,
41669
- style: {
41670
- display: 'block'
41671
- },
41727
+ showLegend: options.showLegend,
41672
41728
  children: [jsxRuntime.jsxs(group.Group, {
41673
41729
  left: margin.left,
41674
41730
  top: margin.top,
41675
- children: [jsxRuntime.jsx(grid.GridRows, {
41676
- tickValues: yTickValues.length > 0 ? yTickValues : undefined,
41677
- scale: yScale,
41731
+ children: [jsxRuntime.jsx(GridRows, {
41732
+ ticks: yTickValues,
41733
+ yScale: yScale,
41678
41734
  width: innerWidth,
41679
41735
  height: innerHeight,
41680
- pointerEvents: "none",
41681
- strokeDasharray: "0.5 5",
41682
- strokeWidth: options.removeStroke ? 0 : 2,
41683
- lineStyle: {
41684
- strokeLinecap: 'round',
41685
- stroke: themeCSS.grid.stroke
41686
- }
41687
- }), options.axis.showXAxisLabels && jsxRuntime.jsx(axis.AxisBottom, {
41688
- label: chart.x.title || undefined,
41689
- labelProps: {
41690
- style: AXIS_TITLE_STYLES$1
41691
- },
41692
- labelOffset: margin.bottomTitleOffset,
41693
- hideTicks: true,
41694
- top: innerHeight
41695
- // @ts-ignore
41696
- ,
41697
- scale: xScale,
41698
- tickFormat: function tickFormat(value) {
41699
- var tick = null;
41700
- if (xScaleDataType === 'date_time' && value instanceof Date) {
41701
- // if chart.x.scale.key !== chart.x.key, use tickValue.scaleValue NOT tickValuevalue
41702
- var matchingTickValue = chart.x.ticks.find(function (tickValue) {
41703
- return tickValue.scaleValue && new Date(tickValue.scaleValue).valueOf() === value.valueOf();
41704
- });
41705
- tick = matchingTickValue || null;
41706
- } else {
41707
- var _chart$x;
41708
- var _matchingTickValue = chart == null || (_chart$x = chart.x) == null ? void 0 : _chart$x.ticks.find(function (tickValue) {
41709
- return ((tickValue == null ? void 0 : tickValue.scaleValue) !== null ? tickValue == null ? void 0 : tickValue.scaleValue : 0) === value;
41710
- });
41711
- tick = _matchingTickValue || null;
41712
- }
41713
- if (tick) {
41714
- if (tick.formattedValue) {
41715
- return tick.formattedValue;
41716
- } else {
41717
- return tick.value.toString();
41718
- }
41719
- } else {
41720
- return ''; // ?
41721
- }
41722
- },
41723
- 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) {
41724
- return (tick == null ? void 0 : tick.scaleValue) !== null ? tick == null ? void 0 : tick.scaleValue : 0;
41725
- }) : undefined,
41726
- tickComponent: function tickComponent(_ref2) {
41727
- var formattedValue = _ref2.formattedValue,
41728
- tickProps = _objectWithoutPropertiesLoose(_ref2, _excluded$f);
41729
- return jsxRuntime.jsx(text$3.Text, _extends({
41730
- style: themeCSS.labels
41731
- }, tickProps, {
41732
- children: formattedValue
41733
- }));
41734
- },
41735
- stroke: themeCSS.axis.stroke,
41736
- strokeWidth: options.removeStroke ? 0 : 1
41737
- }), options.axis.showYAxisLabels && jsxRuntime.jsx(axis.AxisLeft, {
41738
- labelOffset: margin.leftTitleOffset,
41739
- label: chart.y.title || undefined,
41740
- labelProps: {
41741
- style: AXIS_TITLE_STYLES$1
41742
- },
41743
- hideTicks: true,
41744
- left: 0,
41745
- top: 0,
41746
- scale: yScale,
41747
- tickFormat: function tickFormat(value) {
41748
- var item = chart.y.ticks.filter(function (tick) {
41749
- return tick.value === value;
41750
- })[0];
41751
- if (item) {
41752
- if (item.formattedValue) {
41753
- return item.formattedValue;
41754
- } else {
41755
- return item.value.toString();
41756
- }
41757
- } else {
41758
- return '';
41759
- }
41760
- },
41761
- tickValues: yTickValues,
41762
- tickComponent: function tickComponent(_ref3) {
41763
- var formattedValue = _ref3.formattedValue,
41764
- tickProps = _objectWithoutPropertiesLoose(_ref3, _excluded2$5);
41765
- return jsxRuntime.jsx(text$3.Text, _extends({
41766
- width: 10,
41767
- style: themeCSS.labels
41768
- }, tickProps, {
41769
- children: formattedValue
41770
- }));
41771
- },
41772
- 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
41773
41755
  }), jsxRuntime.jsxs(group.Group, {
41774
41756
  children: [!options.stacked && jsxRuntime.jsx(shape.BarGroup, {
41775
41757
  data: dataFlattened,
@@ -41869,7 +41851,14 @@ var BarChart$5 = function BarChart(_ref) {
41869
41851
  pointerEvents: "none",
41870
41852
  opacity: 0.8
41871
41853
  })
41872
- }), 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
+ })]
41873
41862
  }), options.showLegend && jsxRuntime.jsx(Legend$1, {
41874
41863
  legendItems: chart.bars,
41875
41864
  visibleYKeys: chart.bars.map(function (legendItem) {
@@ -41917,8 +41906,8 @@ var Bar = function Bar(props) {
41917
41906
  className: props.enableHover ? /*#__PURE__*/css$1.css( {
41918
41907
  name: "4nk3o1-Bar",
41919
41908
  styles: "cursor:pointer;label:Bar;",
41920
- 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"]} */",
41921
- 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
41922
41911
  }) : ''
41923
41912
  }, props.key);
41924
41913
  };