@rivet-health/design-system 5.2.2 → 5.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (31) hide show
  1. package/esm2020/lib/input/calendar/calendar.component.mjs +1 -1
  2. package/esm2020/lib/input/date-range/date-range.component.mjs +1 -1
  3. package/esm2020/lib/input/input-label/input-label.component.mjs +1 -1
  4. package/esm2020/lib/input/single-select/single-select.component.mjs +20 -44
  5. package/esm2020/lib/interaction/focus-on-init.directive.mjs +43 -0
  6. package/esm2020/lib/modal/help/help.component.mjs +11 -3
  7. package/esm2020/lib/riv.module.mjs +11 -1
  8. package/esm2020/lib/size/content.service.mjs +31 -0
  9. package/esm2020/lib/size/size.directive.mjs +40 -0
  10. package/esm2020/lib/visualization/stacked-column/stacked-column.component.mjs +63 -39
  11. package/esm2020/lib/visualization/stacked-column/stacked-column.helpers.mjs +1 -10
  12. package/esm2020/lib/visualization/stacked-row/stacked-row.component.mjs +53 -78
  13. package/esm2020/lib/visualization/time-series/time-series.component.mjs +22 -27
  14. package/esm2020/public-api.mjs +3 -1
  15. package/fesm2015/rivet-health-design-system.mjs +268 -183
  16. package/fesm2015/rivet-health-design-system.mjs.map +1 -1
  17. package/fesm2020/rivet-health-design-system.mjs +267 -188
  18. package/fesm2020/rivet-health-design-system.mjs.map +1 -1
  19. package/lib/input/single-select/single-select.component.d.ts +3 -6
  20. package/lib/interaction/focus-on-init.directive.d.ts +13 -0
  21. package/lib/modal/help/help.component.d.ts +6 -1
  22. package/lib/riv.module.d.ts +31 -29
  23. package/lib/size/content.service.d.ts +16 -0
  24. package/lib/size/size.directive.d.ts +20 -0
  25. package/lib/visualization/stacked-column/stacked-column.component.d.ts +11 -6
  26. package/lib/visualization/stacked-column/stacked-column.helpers.d.ts +0 -1
  27. package/lib/visualization/stacked-row/stacked-row.component.d.ts +9 -13
  28. package/lib/visualization/time-series/time-series.component.d.ts +5 -6
  29. package/package.json +1 -1
  30. package/public-api.d.ts +2 -0
  31. package/styles/tokens/typography.css +1 -1
@@ -5,15 +5,15 @@ import { stack } from 'd3-shape';
5
5
  import { utcFormat } from 'd3-time-format';
6
6
  import { BehaviorSubject, combineLatest, map, shareReplay, } from 'rxjs';
7
7
  import { getIntervalTitle, getMajorInterval, getMajorIntervalFormat, getMinorIntervalFormat, getTimeInterval, intervals, } from '../intervals';
8
- import { getColumnWidth, getDateRange, pointReducer, } from './stacked-column.helpers';
8
+ import { getDateRange, pointReducer } from './stacked-column.helpers';
9
9
  import * as i0 from "@angular/core";
10
10
  import * as i1 from "@angular/common";
11
11
  import * as i2 from "../../modal/callout/callout.component";
12
12
  import * as i3 from "../../modal/callout/callout.directive";
13
13
  import * as i4 from "../../input/single-select/single-select.component";
14
- import * as i5 from "../zero-state/zero-state.component";
15
- import * as i6 from "../../format/pipes/option-group.pipe";
16
- const MAX_COLUMNS = 72;
14
+ import * as i5 from "../../size/size.directive";
15
+ import * as i6 from "../zero-state/zero-state.component";
16
+ import * as i7 from "../../format/pipes/option-group.pipe";
17
17
  const DEFAULT_INTERVAL = 'month';
18
18
  // TODO: once we upgrade to Angular 16, this component can be cleaned up with
19
19
  // signals instead of RxJS.
@@ -21,16 +21,30 @@ export class StackedColumnComponent {
21
21
  constructor() {
22
22
  this.input$ = new BehaviorSubject([]);
23
23
  this.width$ = new BehaviorSubject(960);
24
- this.height$ = new BehaviorSubject(256);
24
+ this.HEIGHT = 300;
25
25
  this.valueFormatter = v => v.toString();
26
26
  this.interval$ = new BehaviorSubject(DEFAULT_INTERVAL);
27
- this.allowedIntervals$ = this.input$.pipe(map(input => {
28
- const numVisibleColumns = input.filter(stack => stack.filter(s => s.style === 'tooltipOnly').length === 0).length;
27
+ this.COLUMN_WIDTHS = [8, 16, 40, 72];
28
+ this.MIN_COLUMN_WIDTH = this.COLUMN_WIDTHS[0];
29
+ this.numInvisibleColumns$ = this.input$.pipe(map(input => input.filter(stack => stack.filter(s => s.style == 'tooltipOnly').length > 0).length));
30
+ this.numVisibleColumns$ = combineLatest([
31
+ this.input$,
32
+ this.numInvisibleColumns$,
33
+ ]).pipe(map(([input, numInvisibleColumns]) => input.length - numInvisibleColumns));
34
+ this.maxColumns$ = this.width$.pipe(map(width => {
35
+ const innerWidth = width - this.LEFT_OFFSET;
36
+ const columnWidthWithPadding = this.MIN_COLUMN_WIDTH + this.COLUMN_PADDING * 2;
37
+ return Math.floor(innerWidth / columnWidthWithPadding);
38
+ }));
39
+ this.allowedIntervals$ = combineLatest([
40
+ this.input$,
41
+ this.maxColumns$,
42
+ this.numVisibleColumns$,
43
+ ]).pipe(map(([input, maxColumns, numVisibleColumns]) => {
29
44
  const [minDate, maxDate] = getDateRange(input);
30
45
  return intervals.filter(interval => {
31
- const timeInterval = getTimeInterval(interval);
32
- const xStepsCount = timeInterval.range(timeInterval.floor(minDate), timeInterval.floor(timeInterval.offset(maxDate, 1))).length;
33
- return numVisibleColumns * xStepsCount <= MAX_COLUMNS;
46
+ const xStepsCount = this.getXStepsInRange(interval, minDate, maxDate);
47
+ return numVisibleColumns * xStepsCount <= maxColumns;
34
48
  });
35
49
  }), shareReplay({ refCount: true, bufferSize: 1 }));
36
50
  this.intervalOptions$ = this.allowedIntervals$.pipe(map(intervals => intervals.map((interval) => ({
@@ -55,6 +69,26 @@ export class StackedColumnComponent {
55
69
  return options.find(o => o.id === interval) ?? null;
56
70
  }), shareReplay({ refCount: true, bufferSize: 1 }));
57
71
  this.selectedInterval$ = this.selectedIntervalOption$.pipe(map(option => (option ? option.id : DEFAULT_INTERVAL)));
72
+ this.columnWidth$ = combineLatest([
73
+ this.input$,
74
+ this.selectedInterval$,
75
+ this.numVisibleColumns$,
76
+ this.width$,
77
+ ]).pipe(map(([input, selectedInterval, numVisibleColumns, width]) => {
78
+ const [minDate, maxDate] = getDateRange(input);
79
+ const xStepCount = this.getXStepsInRange(selectedInterval, minDate, maxDate);
80
+ const innerWidth = width - this.LEFT_OFFSET;
81
+ const allowedColumnWidths = this.COLUMN_WIDTHS.filter(columnWidth => {
82
+ const columnWidthWithPadding = columnWidth + this.COLUMN_PADDING * 2;
83
+ return (innerWidth -
84
+ numVisibleColumns * xStepCount * columnWidthWithPadding >=
85
+ 0);
86
+ });
87
+ const columnWidth = allowedColumnWidths.at(-1);
88
+ if (!columnWidth)
89
+ return 0;
90
+ return columnWidth;
91
+ }));
58
92
  this.binnedData$ = combineLatest([
59
93
  this.input$,
60
94
  this.selectedInterval$,
@@ -79,16 +113,16 @@ export class StackedColumnComponent {
79
113
  });
80
114
  })), shareReplay({ refCount: true, bufferSize: 1 }));
81
115
  this.LEFT_OFFSET = 48;
116
+ this.COLUMN_PADDING = 4;
82
117
  this.drawData$ = combineLatest([
83
118
  this.binnedData$,
84
119
  this.width$,
85
- this.height$,
86
120
  this.selectedInterval$,
87
- ]).pipe(map(([binnedData, width, height, interval]) => {
88
- const viewBox = `0 0 ${width} ${height}`;
121
+ this.numInvisibleColumns$,
122
+ this.columnWidth$,
123
+ ]).pipe(map(([binnedData, width, interval, numInvisibleColumns, columnWidth]) => {
124
+ const viewBox = `0 0 ${width} ${this.HEIGHT}`;
89
125
  const padding = 16;
90
- const invisColumns = binnedData.filter(stack => stack.filter(s => s.style == 'tooltipOnly').length > 0).length;
91
- const visibleColumns = binnedData.length - invisColumns;
92
126
  const [minDate, maxDate] = getDateRange(binnedData);
93
127
  const hasMultipleYears = minDate.getFullYear() < maxDate.getFullYear();
94
128
  const xMinorIntervalFormat = getMinorIntervalFormat(interval);
@@ -102,11 +136,9 @@ export class StackedColumnComponent {
102
136
  const xOuterScale = scaleBand()
103
137
  .domain(xSteps.map(date => fullFormatter(date)))
104
138
  .range([this.LEFT_OFFSET, width]);
105
- const columnWidth = getColumnWidth(xSteps.length * visibleColumns);
106
- const columnPadding = 4;
107
139
  const outerPadding = (xOuterScale.bandwidth() -
108
- (columnPadding * (binnedData.length - 1 - invisColumns) +
109
- columnWidth * (binnedData.length - invisColumns))) /
140
+ (this.COLUMN_PADDING * (binnedData.length - 1 - numInvisibleColumns) +
141
+ columnWidth * (binnedData.length - numInvisibleColumns))) /
110
142
  2;
111
143
  const renderInformation = binnedData.map(stackData => {
112
144
  /*
@@ -184,7 +216,7 @@ export class StackedColumnComponent {
184
216
  domainMax = 1;
185
217
  const yScale = scaleLinear()
186
218
  .domain([0, domainMax])
187
- .range([height - padding * 2, padding]);
219
+ .range([this.HEIGHT - padding * 2, padding]);
188
220
  const rects = [];
189
221
  let invisColumnsSoFar = 0;
190
222
  // For each unique stack,
@@ -204,7 +236,7 @@ export class StackedColumnComponent {
204
236
  const x = (xOuterScale(fullFormatter(date)) ?? 0) +
205
237
  outerPadding +
206
238
  columnWidth * (index - invisColumnsSoFar) +
207
- columnPadding * (index - invisColumnsSoFar);
239
+ this.COLUMN_PADDING * (index - invisColumnsSoFar);
208
240
  const y = yScale(baseline) - height;
209
241
  let fill = '';
210
242
  if (seriesPoint.series.style != 'tooltipOnly') {
@@ -242,8 +274,9 @@ export class StackedColumnComponent {
242
274
  .filter((tick, i, ticks) => i === 0 || tick.label !== ticks[i - 1].label)
243
275
  : [];
244
276
  const yTicks = yScale.ticks(5);
245
- const hoverBandWidth = (renderInformation.length - invisColumns) * columnWidth +
246
- (renderInformation.length + 1 - invisColumns) * columnPadding;
277
+ const hoverBandWidth = (renderInformation.length - numInvisibleColumns) * columnWidth +
278
+ (renderInformation.length + 1 - numInvisibleColumns) *
279
+ this.COLUMN_PADDING;
247
280
  const hoverBands = xSteps.map(date => ({
248
281
  dateValue: date.valueOf(),
249
282
  x: (xOuterScale(fullFormatter(date)) ?? 0) +
@@ -290,36 +323,27 @@ export class StackedColumnComponent {
290
323
  set width(v) {
291
324
  this.width$.next(v);
292
325
  }
293
- get width() {
294
- return this.width$.getValue();
295
- }
296
- set height(v) {
297
- this.height$.next(v);
298
- }
299
- get height() {
300
- return this.height$.getValue();
301
- }
302
326
  set interval(v) {
303
327
  this.interval$.next(v);
304
328
  }
305
329
  get interval() {
306
330
  return this.interval$.getValue();
307
331
  }
332
+ getXStepsInRange(interval, minDate, maxDate) {
333
+ const timeInterval = getTimeInterval(interval);
334
+ return timeInterval.range(timeInterval.floor(minDate), timeInterval.floor(timeInterval.offset(maxDate, 1))).length;
335
+ }
308
336
  setIntervalOption(option) {
309
337
  this.interval = option.id;
310
338
  }
311
339
  }
312
340
  StackedColumnComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedColumnComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
313
- StackedColumnComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: StackedColumnComponent, selector: "riv-stacked-column", inputs: { input: "input", width: "width", height: "height", valueFormatter: "valueFormatter", interval: "interval" }, viewQueries: [{ propertyName: "controls", first: true, predicate: ["controls"], descendants: true }], ngImport: i0, template: "<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n *ngIf=\"drawData$ | async; let d\"\n [attr.viewBox]=\"d.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"d.columnWidth\"\n [attr.height]=\"d.columnWidth\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n x1=\"0\"\n [attr.y1]=\"d.columnWidth\"\n [attr.x2]=\"d.columnWidth\"\n y2=\"0\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let tick of d.xMinorTicks\">\n <rect\n class=\"tick-background\"\n [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n [attr.x]=\"tick.x - 20\"\n [attr.y]=\"d.yScale(0) + 3\"\n width=\"40\"\n height=\"12\"\n rx=\"2\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 12\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.xMajorTicks\">\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 24\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.yTicks\">\n <rect\n class=\"tick\"\n x=\"0\"\n [attr.y]=\"d.yScale(tick)\"\n width=\"100%\"\n height=\"1\"\n ></rect>\n <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of d.rects\">\n <rect\n class=\"column\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n [attr.fill]=\"rect.fill\"\n [class.blurred]=\"\n (hoveredBand$ | async) !== null &&\n (hoveredBand$ | async)?.dateValue !== rect.dateValue\n \"\n ></rect>\n <rect\n *ngIf=\"rect.striped\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"url(#stripes)\"\n ></rect>\n </g>\n <rect\n *ngFor=\"let rect of d.hoverBands\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"transparent\"\n (mouseenter)=\"\n hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n \"\n (mouseleave)=\"hoveredBand$.next(null)\"\n ></rect>\n </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n <riv-callout\n *riv-callout\n [anchor]=\"callout.anchor\"\n [isModal]=\"false\"\n [preferredPosition]=\"'center-right'\"\n [allowedPositions]=\"[\n 'center-right',\n 'center-left',\n 'top-center',\n 'bottom-center'\n ]\"\n >\n <div class=\"callout-content\">\n <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n <div>{{ metric.label }}</div>\n <div class=\"callout-metric-value\">{{ metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n <riv-single-select\n [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n [selectedOption]=\"selectedIntervalOption$ | async\"\n (selectedOptionChange)=\"setIntervalOption($event)\"\n ></riv-single-select>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);fill:var(--type-light-low-contrast)}.tick-background{transition:fill var(--short-transition);fill:transparent}.tick-background.focused{fill:var(--baloo-10)}.column{transition:opacity var(--short-transition)}.column.blurred{opacity:.4}.callout-content{padding:var(--size-large);display:grid;gap:var(--size-medium);grid-template-columns:1fr 1fr}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}.callout-metric-value{font-weight:var(--font-weight-heavy)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "directive", type: i3.CalloutDirective, selector: "[riv-callout]" }, { kind: "component", type: i4.SingleSelectComponent, selector: "riv-single-select", inputs: ["groups", "selectedOption", "filterabilityOptions", "loading", "locked", "noOptionsMessage", "nodeTemplate", "triggerTemplate", "placeholder", "disabled"], outputs: ["filterQueryChange", "selectedOptionChange"] }, { kind: "component", type: i5.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.OptionGroupPipe, name: "rivOptionGroup" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
341
+ StackedColumnComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: StackedColumnComponent, selector: "riv-stacked-column", inputs: { input: "input", valueFormatter: "valueFormatter", interval: "interval" }, viewQueries: [{ propertyName: "controls", first: true, predicate: ["controls"], descendants: true }], ngImport: i0, template: "<div\n *ngIf=\"!(empty$ | async); else zeroState\"\n class=\"container\"\n (rivClientSize)=\"width = $event.width\"\n>\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n *ngIf=\"drawData$ | async; let d\"\n [attr.viewBox]=\"d.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"d.columnWidth\"\n [attr.height]=\"d.columnWidth\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n x1=\"0\"\n [attr.y1]=\"d.columnWidth\"\n [attr.x2]=\"d.columnWidth\"\n y2=\"0\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let tick of d.xMinorTicks\">\n <rect\n class=\"tick-background\"\n [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n [attr.x]=\"tick.x - 20\"\n [attr.y]=\"d.yScale(0) + 3\"\n width=\"40\"\n height=\"12\"\n rx=\"2\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 12\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.xMajorTicks\">\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 24\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.yTicks\">\n <rect\n class=\"tick\"\n x=\"0\"\n [attr.y]=\"d.yScale(tick)\"\n width=\"100%\"\n height=\"1\"\n ></rect>\n <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of d.rects\">\n <rect\n class=\"column\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n [attr.fill]=\"rect.fill\"\n [class.blurred]=\"\n (hoveredBand$ | async) !== null &&\n (hoveredBand$ | async)?.dateValue !== rect.dateValue\n \"\n ></rect>\n <rect\n *ngIf=\"rect.striped\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"url(#stripes)\"\n ></rect>\n </g>\n <rect\n *ngFor=\"let rect of d.hoverBands\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"transparent\"\n (mouseenter)=\"\n hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n \"\n (mouseleave)=\"hoveredBand$.next(null)\"\n ></rect>\n </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n <riv-callout\n *riv-callout\n [anchor]=\"callout.anchor\"\n [isModal]=\"false\"\n [preferredPosition]=\"'center-right'\"\n [allowedPositions]=\"[\n 'center-right',\n 'center-left',\n 'top-center',\n 'bottom-center'\n ]\"\n >\n <div class=\"callout-content\">\n <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n <div>{{ metric.label }}</div>\n <div class=\"callout-metric-value\">{{ metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n <riv-single-select\n [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n [selectedOption]=\"selectedIntervalOption$ | async\"\n (selectedOptionChange)=\"setIntervalOption($event)\"\n ></riv-single-select>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);fill:var(--type-light-low-contrast)}.tick-background{transition:fill var(--short-transition);fill:transparent}.tick-background.focused{fill:var(--baloo-10)}.column{transition:opacity var(--short-transition)}.column.blurred{opacity:.4}.callout-content{padding:var(--size-large);display:grid;gap:var(--size-medium);grid-template-columns:1fr 1fr}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}.callout-metric-value{font-weight:var(--font-weight-heavy)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "directive", type: i3.CalloutDirective, selector: "[riv-callout]" }, { kind: "component", type: i4.SingleSelectComponent, selector: "riv-single-select", inputs: ["groups", "selectedOption", "filterabilityOptions", "loading", "locked", "maxCalloutHeight", "noOptionsMessage", "nodeTemplate", "triggerTemplate", "placeholder", "disabled"], outputs: ["filterQueryChange", "selectedOptionChange"] }, { kind: "directive", type: i5.SizeDirective, selector: "[rivClientSize]", outputs: ["rivClientSize"] }, { kind: "component", type: i6.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i7.OptionGroupPipe, name: "rivOptionGroup" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
314
342
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedColumnComponent, decorators: [{
315
343
  type: Component,
316
- args: [{ selector: 'riv-stacked-column', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n *ngIf=\"drawData$ | async; let d\"\n [attr.viewBox]=\"d.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"d.columnWidth\"\n [attr.height]=\"d.columnWidth\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n x1=\"0\"\n [attr.y1]=\"d.columnWidth\"\n [attr.x2]=\"d.columnWidth\"\n y2=\"0\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let tick of d.xMinorTicks\">\n <rect\n class=\"tick-background\"\n [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n [attr.x]=\"tick.x - 20\"\n [attr.y]=\"d.yScale(0) + 3\"\n width=\"40\"\n height=\"12\"\n rx=\"2\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 12\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.xMajorTicks\">\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 24\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.yTicks\">\n <rect\n class=\"tick\"\n x=\"0\"\n [attr.y]=\"d.yScale(tick)\"\n width=\"100%\"\n height=\"1\"\n ></rect>\n <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of d.rects\">\n <rect\n class=\"column\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n [attr.fill]=\"rect.fill\"\n [class.blurred]=\"\n (hoveredBand$ | async) !== null &&\n (hoveredBand$ | async)?.dateValue !== rect.dateValue\n \"\n ></rect>\n <rect\n *ngIf=\"rect.striped\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"url(#stripes)\"\n ></rect>\n </g>\n <rect\n *ngFor=\"let rect of d.hoverBands\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"transparent\"\n (mouseenter)=\"\n hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n \"\n (mouseleave)=\"hoveredBand$.next(null)\"\n ></rect>\n </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n <riv-callout\n *riv-callout\n [anchor]=\"callout.anchor\"\n [isModal]=\"false\"\n [preferredPosition]=\"'center-right'\"\n [allowedPositions]=\"[\n 'center-right',\n 'center-left',\n 'top-center',\n 'bottom-center'\n ]\"\n >\n <div class=\"callout-content\">\n <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n <div>{{ metric.label }}</div>\n <div class=\"callout-metric-value\">{{ metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n <riv-single-select\n [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n [selectedOption]=\"selectedIntervalOption$ | async\"\n (selectedOptionChange)=\"setIntervalOption($event)\"\n ></riv-single-select>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);fill:var(--type-light-low-contrast)}.tick-background{transition:fill var(--short-transition);fill:transparent}.tick-background.focused{fill:var(--baloo-10)}.column{transition:opacity var(--short-transition)}.column.blurred{opacity:.4}.callout-content{padding:var(--size-large);display:grid;gap:var(--size-medium);grid-template-columns:1fr 1fr}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}.callout-metric-value{font-weight:var(--font-weight-heavy)}\n"] }]
344
+ args: [{ selector: 'riv-stacked-column', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n *ngIf=\"!(empty$ | async); else zeroState\"\n class=\"container\"\n (rivClientSize)=\"width = $event.width\"\n>\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n *ngIf=\"drawData$ | async; let d\"\n [attr.viewBox]=\"d.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"d.columnWidth\"\n [attr.height]=\"d.columnWidth\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n x1=\"0\"\n [attr.y1]=\"d.columnWidth\"\n [attr.x2]=\"d.columnWidth\"\n y2=\"0\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let tick of d.xMinorTicks\">\n <rect\n class=\"tick-background\"\n [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n [attr.x]=\"tick.x - 20\"\n [attr.y]=\"d.yScale(0) + 3\"\n width=\"40\"\n height=\"12\"\n rx=\"2\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 12\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.xMajorTicks\">\n <text\n class=\"tick-label\"\n [attr.x]=\"tick.x\"\n [attr.y]=\"d.yScale(0) + 24\"\n text-anchor=\"middle\"\n >\n {{ tick.label }}\n </text>\n </g>\n <g *ngFor=\"let tick of d.yTicks\">\n <rect\n class=\"tick\"\n x=\"0\"\n [attr.y]=\"d.yScale(tick)\"\n width=\"100%\"\n height=\"1\"\n ></rect>\n <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of d.rects\">\n <rect\n class=\"column\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n [attr.fill]=\"rect.fill\"\n [class.blurred]=\"\n (hoveredBand$ | async) !== null &&\n (hoveredBand$ | async)?.dateValue !== rect.dateValue\n \"\n ></rect>\n <rect\n *ngIf=\"rect.striped\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"url(#stripes)\"\n ></rect>\n </g>\n <rect\n *ngFor=\"let rect of d.hoverBands\"\n [attr.x]=\"rect.x\"\n [attr.y]=\"rect.y\"\n [attr.width]=\"rect.width\"\n [attr.height]=\"rect.height\"\n fill=\"transparent\"\n (mouseenter)=\"\n hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n \"\n (mouseleave)=\"hoveredBand$.next(null)\"\n ></rect>\n </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n <riv-callout\n *riv-callout\n [anchor]=\"callout.anchor\"\n [isModal]=\"false\"\n [preferredPosition]=\"'center-right'\"\n [allowedPositions]=\"[\n 'center-right',\n 'center-left',\n 'top-center',\n 'bottom-center'\n ]\"\n >\n <div class=\"callout-content\">\n <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n <div>{{ metric.label }}</div>\n <div class=\"callout-metric-value\">{{ metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n <riv-single-select\n [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n [selectedOption]=\"selectedIntervalOption$ | async\"\n (selectedOptionChange)=\"setIntervalOption($event)\"\n ></riv-single-select>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);fill:var(--type-light-low-contrast)}.tick-background{transition:fill var(--short-transition);fill:transparent}.tick-background.focused{fill:var(--baloo-10)}.column{transition:opacity var(--short-transition)}.column.blurred{opacity:.4}.callout-content{padding:var(--size-large);display:grid;gap:var(--size-medium);grid-template-columns:1fr 1fr}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}.callout-metric-value{font-weight:var(--font-weight-heavy)}\n"] }]
317
345
  }], propDecorators: { input: [{
318
346
  type: Input
319
- }], width: [{
320
- type: Input
321
- }], height: [{
322
- type: Input
323
347
  }], valueFormatter: [{
324
348
  type: Input
325
349
  }], interval: [{
@@ -328,4 +352,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
328
352
  type: ViewChild,
329
353
  args: ['controls']
330
354
  }] } });
331
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-column.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { index } from 'd3-array';\nimport { scaleBand, scaleLinear } from 'd3-scale';\nimport { stack } from 'd3-shape';\nimport { utcFormat } from 'd3-time-format';\nimport {\n  BehaviorSubject,\n  Observable,\n  combineLatest,\n  map,\n  shareReplay,\n} from 'rxjs';\nimport { SingleSelectComponent } from '../../input/single-select/single-select.component';\nimport {\n  Interval,\n  getIntervalTitle,\n  getMajorInterval,\n  getMajorIntervalFormat,\n  getMinorIntervalFormat,\n  getTimeInterval,\n  intervals,\n} from '../intervals';\nimport {\n  getColumnWidth,\n  getDateRange,\n  pointReducer,\n} from './stacked-column.helpers';\n\nconst MAX_COLUMNS = 72;\nconst DEFAULT_INTERVAL: Interval = 'month';\n\n// TODO: once we upgrade to Angular 16, this component can be cleaned up with\n// signals instead of RxJS.\n@Component({\n  selector: 'riv-stacked-column',\n  templateUrl: './stacked-column.component.html',\n  styleUrls: ['./stacked-column.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedColumnComponent {\n  private readonly input$ = new BehaviorSubject<StackedColumnComponent.Stack[]>(\n    [],\n  );\n  @Input()\n  public set input(v: StackedColumnComponent.Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): StackedColumnComponent.Stack[] {\n    return this.input$.getValue();\n  }\n\n  private readonly width$ = new BehaviorSubject<number>(960);\n  @Input()\n  public set width(v: number) {\n    this.width$.next(v);\n  }\n  public get width(): number {\n    return this.width$.getValue();\n  }\n\n  private readonly height$ = new BehaviorSubject<number>(256);\n  @Input()\n  public set height(v: number) {\n    this.height$.next(v);\n  }\n  public get height(): number {\n    return this.height$.getValue();\n  }\n\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  private readonly interval$ = new BehaviorSubject<Interval>(DEFAULT_INTERVAL);\n  @Input()\n  public set interval(v: Interval) {\n    this.interval$.next(v);\n  }\n  public get interval() {\n    return this.interval$.getValue();\n  }\n\n  @ViewChild('controls')\n  public readonly controls?: TemplateRef<any>;\n\n  private readonly allowedIntervals$: Observable<Interval[]> = this.input$.pipe(\n    map(input => {\n      const numVisibleColumns = input.filter(\n        stack => stack.filter(s => s.style === 'tooltipOnly').length === 0,\n      ).length;\n\n      const [minDate, maxDate] = getDateRange(input);\n\n      return intervals.filter(interval => {\n        const timeInterval = getTimeInterval(interval);\n        const xStepsCount = timeInterval.range(\n          timeInterval.floor(minDate),\n          timeInterval.floor(timeInterval.offset(maxDate, 1)),\n        ).length;\n        return numVisibleColumns * xStepsCount <= MAX_COLUMNS;\n      });\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly intervalOptions$ = this.allowedIntervals$.pipe(\n    map(intervals =>\n      intervals.map(\n        (interval): SingleSelectComponent.Option => ({\n          id: interval,\n          title: getIntervalTitle(interval),\n        }),\n      ),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly selectedIntervalOption$ = combineLatest([\n    this.interval$,\n    this.intervalOptions$,\n  ]).pipe(\n    map(([interval, options]) => {\n      const allowedIntervals = options.map(({ id }) => id as Interval);\n      if (!allowedIntervals.includes(interval)) {\n        const firstAllowed = allowedIntervals.at(0);\n        const lastAllowed = allowedIntervals.at(-1);\n        if (!firstAllowed || !lastAllowed) return null;\n\n        if (intervals.indexOf(interval) < intervals.indexOf(firstAllowed))\n          interval = firstAllowed;\n        if (intervals.indexOf(interval) > intervals.indexOf(lastAllowed))\n          interval = lastAllowed;\n      }\n      return options.find(o => o.id === interval) ?? null;\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly selectedInterval$ = this.selectedIntervalOption$.pipe(\n    map(option => (option ? (option.id as Interval) : DEFAULT_INTERVAL)),\n  );\n\n  setIntervalOption(option: SingleSelectComponent.Option) {\n    this.interval = option.id as Interval;\n  }\n\n  readonly binnedData$ = combineLatest([\n    this.input$,\n    this.selectedInterval$,\n  ]).pipe(\n    map(([input, interval]) =>\n      input.map((stack): StackedColumnComponent.Stack => {\n        const timeInterval = getTimeInterval(interval);\n\n        return stack.map(series => {\n          const binned = new Map<number, StackedColumnComponent.Value[]>();\n          for (const point of series.data) {\n            const bin = timeInterval.floor(point.date).valueOf();\n            if (!binned.has(bin)) {\n              binned.set(bin, []);\n            }\n            binned.get(bin)!.push(point);\n          }\n          return {\n            ...series,\n            data: [...binned.entries()].map(([dateValue, points]) => ({\n              date: new Date(dateValue),\n              value: pointReducer(points),\n            })),\n          };\n        });\n      }),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly LEFT_OFFSET = 48;\n\n  readonly drawData$ = combineLatest([\n    this.binnedData$,\n    this.width$,\n    this.height$,\n    this.selectedInterval$,\n  ]).pipe(\n    map(([binnedData, width, height, interval]) => {\n      const viewBox = `0 0 ${width} ${height}`;\n      const padding = 16;\n\n      const invisColumns = binnedData.filter(\n        stack => stack.filter(s => s.style == 'tooltipOnly').length > 0,\n      ).length;\n      const visibleColumns = binnedData.length - invisColumns;\n\n      const [minDate, maxDate] = getDateRange(binnedData);\n      const hasMultipleYears = minDate.getFullYear() < maxDate.getFullYear();\n      const xMinorIntervalFormat = getMinorIntervalFormat(interval);\n      const minorFormatter = utcFormat(xMinorIntervalFormat);\n      const minorTimeInterval = getTimeInterval(interval);\n      const xSteps = minorTimeInterval.range(\n        minorTimeInterval.floor(minDate),\n        minorTimeInterval.floor(minorTimeInterval.offset(maxDate, 1)),\n      );\n      const xMajorInterval = getMajorInterval(interval);\n      const xMajorIntervalFormat = getMajorIntervalFormat(\n        xMajorInterval,\n        hasMultipleYears,\n      );\n      const majorFormatter = utcFormat(xMajorIntervalFormat);\n      const fullFormatter = utcFormat(\n        `${xMinorIntervalFormat} ${xMajorIntervalFormat}`,\n      );\n\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => fullFormatter(date)))\n        .range([this.LEFT_OFFSET, width]);\n\n      const columnWidth = getColumnWidth(xSteps.length * visibleColumns);\n      const columnPadding = 4;\n      const outerPadding =\n        (xOuterScale.bandwidth() -\n          (columnPadding * (binnedData.length - 1 - invisColumns) +\n            columnWidth * (binnedData.length - invisColumns))) /\n        2;\n\n      type SeriesPoint = StackedColumnComponent.Value & {\n        series: StackedColumnComponent.Series;\n      };\n\n      const renderInformation = binnedData.map(stackData => {\n        /*\n          Create an array with every data point + their associated series.\n          i.e. Make an backlink for each element to its series.\n        */\n        const stackedTable = stackData.reduce<SeriesPoint[]>(\n          (table, s) => [\n            ...table,\n            ...s.data.map(point => ({\n              ...point,\n              series: s,\n            })),\n          ],\n          [],\n        );\n\n        /*\n          Maps each of the SeriesPoints based on label.\n          Then map each of those maps to a date.\n\n          Example:\n            If we were indexing the following data:\n              * Estimates Complete -> 5 in January, 7 in February\n              * Estimates Incomplete -> 3 in January, 3 in February\n            We would get an indexed table like the following, (note this is pseudocode):\n              'January': { 'Estimates Complete': 5, 'Estimates Incomplete': 3 }\n              'February': { 'Estimates Complete': 7, 'Estimates Incomplete': 3 }\n        */\n        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\n\n        /*\n          The output of \"stack()\" is an array of arrays of tuples.\n            The tuples unnest as [baseline, topline].\n            Each \"key\" involved in the stack has an array of tuples relating to it.\n              i.e. [ Date 1 Tuple, Date 2 Tuple ]\n            All \"key\" arrays are contained in the outer array.\n              i.e. [ Key 1 Tuples, Key 2 Tuples]\n\n          We select the keys using \".keys()\".\n            For our purposes, the keys are the legend elements.\n            i.e. If you have two blocks of the same color, they have the same key.\n\n          We select the heights of the blocks using \".value()\".\n\n          We are going to stack the blocks on top of each other.\n            The baseline of the first key tuples are 0.\n            The baseline of the second key tuples are the topline of the associated first key tuple.\n\n          Example:\n            If we were stacking the following data:\n              * Estimates Complete -> 5 in January, 7 in February\n              * Estimates Incomplete -> 3 in January, 3 in February\n            We would get a stack of:\n              [  [ [0,5],[0,7] ], [ [5,8],[7,10] ]  ]\n        */\n        const stacked = stack<\n          typeof indexedTable extends Map<infer K, infer V> ? [K, V] : never\n        >()\n          .keys(stackData.map(v => v.label))\n          .value(([_, group], key) => {\n            return group.get(key)!.value;\n          })(indexedTable);\n\n        return { indexedTable, stacked };\n      });\n\n      // The domain should be based on the highest value in any stack, on any date.\n      let domainMax = 0;\n      // For each unique stack,\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        // For each date,\n        [...indexedTable.values()].forEach((pointMap, i) => {\n          // For each series,\n          [...pointMap.values()].forEach((_, j) => {\n            if (_.series.style == 'tooltipOnly') {\n              return;\n            }\n            domainMax = Math.max(domainMax, ...stacked[j][i]);\n          });\n        });\n      });\n      if (domainMax === 0) domainMax = 1;\n      const yScale = scaleLinear()\n        .domain([0, domainMax])\n        .range([height - padding * 2, padding]);\n\n      const rects: {\n        dateValue: number;\n        x: number;\n        y: number;\n        width: number;\n        height: number;\n        fill: string;\n        striped: boolean;\n        seriesPoint: SeriesPoint;\n      }[] = [];\n      let invisColumnsSoFar = 0;\n      // For each unique stack,\n      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        let invisColumnFound = false;\n        // For each date,\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          // For each series,\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\n            // Get the size of the box we are about to draw.\n            const [baseline, topline] = stacked[j][i];\n\n            if (seriesPoint.series.style == 'tooltipOnly') {\n              invisColumnFound = true;\n            }\n            const width =\n              seriesPoint.series.style == 'tooltipOnly' ? 0 : columnWidth;\n            const height = yScale(baseline) - yScale(topline);\n            const x =\n              (xOuterScale(fullFormatter(date)) ?? 0) +\n              outerPadding +\n              columnWidth * (index - invisColumnsSoFar) +\n              columnPadding * (index - invisColumnsSoFar);\n            const y = yScale(baseline) - height;\n            let fill = '';\n            if (seriesPoint.series.style != 'tooltipOnly') {\n              fill = `var(${seriesPoint.series.colorToken})`;\n            }\n\n            rects.push({\n              dateValue: date.valueOf(),\n              x,\n              y,\n              width,\n              height,\n              fill,\n              striped: seriesPoint.series.style === 'striped',\n              seriesPoint,\n            });\n          });\n        });\n        if (invisColumnFound) {\n          invisColumnsSoFar++;\n        }\n      });\n\n      const xMinorTicks = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        label: minorFormatter(date),\n        x:\n          (xOuterScale(fullFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,\n      }));\n      const xMajorTicks =\n        interval !== 'year'\n          ? xSteps\n              .map(date => ({\n                dateValue: date.valueOf(),\n                label: majorFormatter(date),\n                x:\n                  (xOuterScale(fullFormatter(date)) ?? 0) +\n                  xOuterScale.bandwidth() / 2,\n              }))\n              .filter(\n                (tick, i, ticks) =>\n                  i === 0 || tick.label !== ticks[i - 1].label,\n              )\n          : [];\n\n      const yTicks = yScale.ticks(5);\n\n      const hoverBandWidth =\n        (renderInformation.length - invisColumns) * columnWidth +\n        (renderInformation.length + 1 - invisColumns) * columnPadding;\n      const hoverBands = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        x:\n          (xOuterScale(fullFormatter(date)) ?? 0) +\n          (xOuterScale.bandwidth() - hoverBandWidth) / 2,\n        y: yScale(domainMax),\n        width: hoverBandWidth,\n        height: yScale(0) - yScale(domainMax),\n      }));\n\n      return {\n        columnWidth,\n        hoverBands,\n        rects,\n        viewBox,\n        xMinorTicks,\n        xMajorTicks,\n        yScale,\n        yTicks,\n      };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly empty$ = this.input$.pipe(\n    map(input =>\n      input.every(stack => stack.every(series => series.data.length === 0)),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly hoveredBand$ = new BehaviorSubject<{\n    dateValue: number;\n    event: MouseEvent;\n  } | null>(null);\n\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand) return null;\n\n      const { target } = hoveredBand.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      type Metric = { label: string; value: string };\n      const metrics: Metric[] = drawData.rects\n        .filter(rect => rect.dateValue === hoveredBand.dateValue)\n        .map(rect => ({\n          label: rect.seriesPoint.series.label,\n          value: this.valueFormatter(rect.seriesPoint.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n\nexport namespace StackedColumnComponent {\n  export type Value = { date: Date; value: number };\n\n  export type Series = {\n    label: string;\n    data: Value[];\n  } & (\n    | {\n        colorToken: string;\n        style?: 'solid' | 'striped';\n      }\n    | {\n        style: 'tooltipOnly';\n      }\n  );\n\n  export type Stack = Series[];\n}\n","<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n  <svg\n    xmlns=\"http://www.w3.org/2000/svg\"\n    *ngIf=\"drawData$ | async; let d\"\n    [attr.viewBox]=\"d.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"d.columnWidth\"\n        [attr.height]=\"d.columnWidth\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          x1=\"0\"\n          [attr.y1]=\"d.columnWidth\"\n          [attr.x2]=\"d.columnWidth\"\n          y2=\"0\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let tick of d.xMinorTicks\">\n      <rect\n        class=\"tick-background\"\n        [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n        [attr.x]=\"tick.x - 20\"\n        [attr.y]=\"d.yScale(0) + 3\"\n        width=\"40\"\n        height=\"12\"\n        rx=\"2\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 12\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.xMajorTicks\">\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 24\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.yTicks\">\n      <rect\n        class=\"tick\"\n        x=\"0\"\n        [attr.y]=\"d.yScale(tick)\"\n        width=\"100%\"\n        height=\"1\"\n      ></rect>\n      <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g *ngFor=\"let rect of d.rects\">\n      <rect\n        class=\"column\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        [attr.fill]=\"rect.fill\"\n        [class.blurred]=\"\n          (hoveredBand$ | async) !== null &&\n          (hoveredBand$ | async)?.dateValue !== rect.dateValue\n        \"\n      ></rect>\n      <rect\n        *ngIf=\"rect.striped\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        fill=\"url(#stripes)\"\n      ></rect>\n    </g>\n    <rect\n      *ngFor=\"let rect of d.hoverBands\"\n      [attr.x]=\"rect.x\"\n      [attr.y]=\"rect.y\"\n      [attr.width]=\"rect.width\"\n      [attr.height]=\"rect.height\"\n      fill=\"transparent\"\n      (mouseenter)=\"\n        hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n      \"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    ></rect>\n  </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n  <riv-callout\n    *riv-callout\n    [anchor]=\"callout.anchor\"\n    [isModal]=\"false\"\n    [preferredPosition]=\"'center-right'\"\n    [allowedPositions]=\"[\n      'center-right',\n      'center-left',\n      'top-center',\n      'bottom-center'\n    ]\"\n  >\n    <div class=\"callout-content\">\n      <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n        <div>{{ metric.label }}</div>\n        <div class=\"callout-metric-value\">{{ metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n  <riv-single-select\n    [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n    [selectedOption]=\"selectedIntervalOption$ | async\"\n    (selectedOptionChange)=\"setIntervalOption($event)\"\n  ></riv-single-select>\n</ng-template>\n"]}
355
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-column.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { index } from 'd3-array';\nimport { scaleBand, scaleLinear } from 'd3-scale';\nimport { stack } from 'd3-shape';\nimport { utcFormat } from 'd3-time-format';\nimport {\n  BehaviorSubject,\n  Observable,\n  combineLatest,\n  map,\n  shareReplay,\n} from 'rxjs';\nimport { SingleSelectComponent } from '../../input/single-select/single-select.component';\nimport {\n  Interval,\n  getIntervalTitle,\n  getMajorInterval,\n  getMajorIntervalFormat,\n  getMinorIntervalFormat,\n  getTimeInterval,\n  intervals,\n} from '../intervals';\nimport { getDateRange, pointReducer } from './stacked-column.helpers';\n\nconst DEFAULT_INTERVAL: Interval = 'month';\n\n// TODO: once we upgrade to Angular 16, this component can be cleaned up with\n// signals instead of RxJS.\n@Component({\n  selector: 'riv-stacked-column',\n  templateUrl: './stacked-column.component.html',\n  styleUrls: ['./stacked-column.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedColumnComponent {\n  private readonly input$ = new BehaviorSubject<StackedColumnComponent.Stack[]>(\n    [],\n  );\n  @Input()\n  public set input(v: StackedColumnComponent.Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): StackedColumnComponent.Stack[] {\n    return this.input$.getValue();\n  }\n\n  private readonly width$ = new BehaviorSubject<number>(960);\n  protected set width(v: number) {\n    this.width$.next(v);\n  }\n\n  private readonly HEIGHT = 300;\n\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  private readonly interval$ = new BehaviorSubject<Interval>(DEFAULT_INTERVAL);\n  @Input()\n  public set interval(v: Interval) {\n    this.interval$.next(v);\n  }\n  public get interval() {\n    return this.interval$.getValue();\n  }\n\n  @ViewChild('controls')\n  public readonly controls?: TemplateRef<any>;\n\n  private readonly COLUMN_WIDTHS = [8, 16, 40, 72] as const;\n  private readonly MIN_COLUMN_WIDTH = this.COLUMN_WIDTHS[0];\n\n  private readonly numInvisibleColumns$ = this.input$.pipe(\n    map(\n      input =>\n        input.filter(\n          stack => stack.filter(s => s.style == 'tooltipOnly').length > 0,\n        ).length,\n    ),\n  );\n  private readonly numVisibleColumns$ = combineLatest([\n    this.input$,\n    this.numInvisibleColumns$,\n  ]).pipe(\n    map(([input, numInvisibleColumns]) => input.length - numInvisibleColumns),\n  );\n\n  private readonly maxColumns$ = this.width$.pipe(\n    map(width => {\n      const innerWidth = width - this.LEFT_OFFSET;\n      const columnWidthWithPadding =\n        this.MIN_COLUMN_WIDTH + this.COLUMN_PADDING * 2;\n      return Math.floor(innerWidth / columnWidthWithPadding);\n    }),\n  );\n\n  getXStepsInRange(interval: Interval, minDate: Date, maxDate: Date) {\n    const timeInterval = getTimeInterval(interval);\n    return timeInterval.range(\n      timeInterval.floor(minDate),\n      timeInterval.floor(timeInterval.offset(maxDate, 1)),\n    ).length;\n  }\n\n  private readonly allowedIntervals$: Observable<Interval[]> = combineLatest([\n    this.input$,\n    this.maxColumns$,\n    this.numVisibleColumns$,\n  ]).pipe(\n    map(([input, maxColumns, numVisibleColumns]) => {\n      const [minDate, maxDate] = getDateRange(input);\n\n      return intervals.filter(interval => {\n        const xStepsCount = this.getXStepsInRange(interval, minDate, maxDate);\n        return numVisibleColumns * xStepsCount <= maxColumns;\n      });\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly intervalOptions$ = this.allowedIntervals$.pipe(\n    map(intervals =>\n      intervals.map(\n        (interval): SingleSelectComponent.Option => ({\n          id: interval,\n          title: getIntervalTitle(interval),\n        }),\n      ),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly selectedIntervalOption$ = combineLatest([\n    this.interval$,\n    this.intervalOptions$,\n  ]).pipe(\n    map(([interval, options]) => {\n      const allowedIntervals = options.map(({ id }) => id as Interval);\n      if (!allowedIntervals.includes(interval)) {\n        const firstAllowed = allowedIntervals.at(0);\n        const lastAllowed = allowedIntervals.at(-1);\n        if (!firstAllowed || !lastAllowed) return null;\n\n        if (intervals.indexOf(interval) < intervals.indexOf(firstAllowed))\n          interval = firstAllowed;\n        if (intervals.indexOf(interval) > intervals.indexOf(lastAllowed))\n          interval = lastAllowed;\n      }\n      return options.find(o => o.id === interval) ?? null;\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly selectedInterval$ = this.selectedIntervalOption$.pipe(\n    map(option => (option ? (option.id as Interval) : DEFAULT_INTERVAL)),\n  );\n\n  setIntervalOption(option: SingleSelectComponent.Option) {\n    this.interval = option.id as Interval;\n  }\n\n  private columnWidth$ = combineLatest([\n    this.input$,\n    this.selectedInterval$,\n    this.numVisibleColumns$,\n    this.width$,\n  ]).pipe(\n    map(([input, selectedInterval, numVisibleColumns, width]) => {\n      const [minDate, maxDate] = getDateRange(input);\n      const xStepCount = this.getXStepsInRange(\n        selectedInterval,\n        minDate,\n        maxDate,\n      );\n\n      const innerWidth = width - this.LEFT_OFFSET;\n      const allowedColumnWidths = this.COLUMN_WIDTHS.filter(columnWidth => {\n        const columnWidthWithPadding = columnWidth + this.COLUMN_PADDING * 2;\n        return (\n          innerWidth -\n            numVisibleColumns * xStepCount * columnWidthWithPadding >=\n          0\n        );\n      });\n      const columnWidth = allowedColumnWidths.at(-1);\n      if (!columnWidth) return 0;\n\n      return columnWidth;\n    }),\n  );\n\n  readonly binnedData$ = combineLatest([\n    this.input$,\n    this.selectedInterval$,\n  ]).pipe(\n    map(([input, interval]) =>\n      input.map((stack): StackedColumnComponent.Stack => {\n        const timeInterval = getTimeInterval(interval);\n\n        return stack.map(series => {\n          const binned = new Map<number, StackedColumnComponent.Value[]>();\n          for (const point of series.data) {\n            const bin = timeInterval.floor(point.date).valueOf();\n            if (!binned.has(bin)) {\n              binned.set(bin, []);\n            }\n            binned.get(bin)!.push(point);\n          }\n          return {\n            ...series,\n            data: [...binned.entries()].map(([dateValue, points]) => ({\n              date: new Date(dateValue),\n              value: pointReducer(points),\n            })),\n          };\n        });\n      }),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly LEFT_OFFSET = 48;\n  readonly COLUMN_PADDING = 4;\n\n  readonly drawData$ = combineLatest([\n    this.binnedData$,\n    this.width$,\n    this.selectedInterval$,\n    this.numInvisibleColumns$,\n    this.columnWidth$,\n  ]).pipe(\n    map(([binnedData, width, interval, numInvisibleColumns, columnWidth]) => {\n      const viewBox = `0 0 ${width} ${this.HEIGHT}`;\n      const padding = 16;\n\n      const [minDate, maxDate] = getDateRange(binnedData);\n      const hasMultipleYears = minDate.getFullYear() < maxDate.getFullYear();\n      const xMinorIntervalFormat = getMinorIntervalFormat(interval);\n      const minorFormatter = utcFormat(xMinorIntervalFormat);\n      const minorTimeInterval = getTimeInterval(interval);\n      const xSteps = minorTimeInterval.range(\n        minorTimeInterval.floor(minDate),\n        minorTimeInterval.floor(minorTimeInterval.offset(maxDate, 1)),\n      );\n      const xMajorInterval = getMajorInterval(interval);\n      const xMajorIntervalFormat = getMajorIntervalFormat(\n        xMajorInterval,\n        hasMultipleYears,\n      );\n      const majorFormatter = utcFormat(xMajorIntervalFormat);\n      const fullFormatter = utcFormat(\n        `${xMinorIntervalFormat} ${xMajorIntervalFormat}`,\n      );\n\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => fullFormatter(date)))\n        .range([this.LEFT_OFFSET, width]);\n\n      const outerPadding =\n        (xOuterScale.bandwidth() -\n          (this.COLUMN_PADDING * (binnedData.length - 1 - numInvisibleColumns) +\n            columnWidth * (binnedData.length - numInvisibleColumns))) /\n        2;\n\n      type SeriesPoint = StackedColumnComponent.Value & {\n        series: StackedColumnComponent.Series;\n      };\n\n      const renderInformation = binnedData.map(stackData => {\n        /*\n          Create an array with every data point + their associated series.\n          i.e. Make an backlink for each element to its series.\n        */\n        const stackedTable = stackData.reduce<SeriesPoint[]>(\n          (table, s) => [\n            ...table,\n            ...s.data.map(point => ({\n              ...point,\n              series: s,\n            })),\n          ],\n          [],\n        );\n\n        /*\n          Maps each of the SeriesPoints based on label.\n          Then map each of those maps to a date.\n\n          Example:\n            If we were indexing the following data:\n              * Estimates Complete -> 5 in January, 7 in February\n              * Estimates Incomplete -> 3 in January, 3 in February\n            We would get an indexed table like the following, (note this is pseudocode):\n              'January': { 'Estimates Complete': 5, 'Estimates Incomplete': 3 }\n              'February': { 'Estimates Complete': 7, 'Estimates Incomplete': 3 }\n        */\n        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\n\n        /*\n          The output of \"stack()\" is an array of arrays of tuples.\n            The tuples unnest as [baseline, topline].\n            Each \"key\" involved in the stack has an array of tuples relating to it.\n              i.e. [ Date 1 Tuple, Date 2 Tuple ]\n            All \"key\" arrays are contained in the outer array.\n              i.e. [ Key 1 Tuples, Key 2 Tuples]\n\n          We select the keys using \".keys()\".\n            For our purposes, the keys are the legend elements.\n            i.e. If you have two blocks of the same color, they have the same key.\n\n          We select the heights of the blocks using \".value()\".\n\n          We are going to stack the blocks on top of each other.\n            The baseline of the first key tuples are 0.\n            The baseline of the second key tuples are the topline of the associated first key tuple.\n\n          Example:\n            If we were stacking the following data:\n              * Estimates Complete -> 5 in January, 7 in February\n              * Estimates Incomplete -> 3 in January, 3 in February\n            We would get a stack of:\n              [  [ [0,5],[0,7] ], [ [5,8],[7,10] ]  ]\n        */\n        const stacked = stack<\n          typeof indexedTable extends Map<infer K, infer V> ? [K, V] : never\n        >()\n          .keys(stackData.map(v => v.label))\n          .value(([_, group], key) => {\n            return group.get(key)!.value;\n          })(indexedTable);\n\n        return { indexedTable, stacked };\n      });\n\n      // The domain should be based on the highest value in any stack, on any date.\n      let domainMax = 0;\n      // For each unique stack,\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        // For each date,\n        [...indexedTable.values()].forEach((pointMap, i) => {\n          // For each series,\n          [...pointMap.values()].forEach((_, j) => {\n            if (_.series.style == 'tooltipOnly') {\n              return;\n            }\n            domainMax = Math.max(domainMax, ...stacked[j][i]);\n          });\n        });\n      });\n      if (domainMax === 0) domainMax = 1;\n      const yScale = scaleLinear()\n        .domain([0, domainMax])\n        .range([this.HEIGHT - padding * 2, padding]);\n\n      const rects: {\n        dateValue: number;\n        x: number;\n        y: number;\n        width: number;\n        height: number;\n        fill: string;\n        striped: boolean;\n        seriesPoint: SeriesPoint;\n      }[] = [];\n      let invisColumnsSoFar = 0;\n      // For each unique stack,\n      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        let invisColumnFound = false;\n        // For each date,\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          // For each series,\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\n            // Get the size of the box we are about to draw.\n            const [baseline, topline] = stacked[j][i];\n\n            if (seriesPoint.series.style == 'tooltipOnly') {\n              invisColumnFound = true;\n            }\n            const width =\n              seriesPoint.series.style == 'tooltipOnly' ? 0 : columnWidth;\n            const height = yScale(baseline) - yScale(topline);\n            const x =\n              (xOuterScale(fullFormatter(date)) ?? 0) +\n              outerPadding +\n              columnWidth * (index - invisColumnsSoFar) +\n              this.COLUMN_PADDING * (index - invisColumnsSoFar);\n            const y = yScale(baseline) - height;\n            let fill = '';\n            if (seriesPoint.series.style != 'tooltipOnly') {\n              fill = `var(${seriesPoint.series.colorToken})`;\n            }\n\n            rects.push({\n              dateValue: date.valueOf(),\n              x,\n              y,\n              width,\n              height,\n              fill,\n              striped: seriesPoint.series.style === 'striped',\n              seriesPoint,\n            });\n          });\n        });\n        if (invisColumnFound) {\n          invisColumnsSoFar++;\n        }\n      });\n\n      const xMinorTicks = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        label: minorFormatter(date),\n        x:\n          (xOuterScale(fullFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,\n      }));\n      const xMajorTicks =\n        interval !== 'year'\n          ? xSteps\n              .map(date => ({\n                dateValue: date.valueOf(),\n                label: majorFormatter(date),\n                x:\n                  (xOuterScale(fullFormatter(date)) ?? 0) +\n                  xOuterScale.bandwidth() / 2,\n              }))\n              .filter(\n                (tick, i, ticks) =>\n                  i === 0 || tick.label !== ticks[i - 1].label,\n              )\n          : [];\n\n      const yTicks = yScale.ticks(5);\n\n      const hoverBandWidth =\n        (renderInformation.length - numInvisibleColumns) * columnWidth +\n        (renderInformation.length + 1 - numInvisibleColumns) *\n          this.COLUMN_PADDING;\n      const hoverBands = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        x:\n          (xOuterScale(fullFormatter(date)) ?? 0) +\n          (xOuterScale.bandwidth() - hoverBandWidth) / 2,\n        y: yScale(domainMax),\n        width: hoverBandWidth,\n        height: yScale(0) - yScale(domainMax),\n      }));\n\n      return {\n        columnWidth,\n        hoverBands,\n        rects,\n        viewBox,\n        xMinorTicks,\n        xMajorTicks,\n        yScale,\n        yTicks,\n      };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly empty$ = this.input$.pipe(\n    map(input =>\n      input.every(stack => stack.every(series => series.data.length === 0)),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly hoveredBand$ = new BehaviorSubject<{\n    dateValue: number;\n    event: MouseEvent;\n  } | null>(null);\n\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand) return null;\n\n      const { target } = hoveredBand.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      type Metric = { label: string; value: string };\n      const metrics: Metric[] = drawData.rects\n        .filter(rect => rect.dateValue === hoveredBand.dateValue)\n        .map(rect => ({\n          label: rect.seriesPoint.series.label,\n          value: this.valueFormatter(rect.seriesPoint.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n\nexport namespace StackedColumnComponent {\n  export type Value = { date: Date; value: number };\n\n  export type Series = {\n    label: string;\n    data: Value[];\n  } & (\n    | {\n        colorToken: string;\n        style?: 'solid' | 'striped';\n      }\n    | {\n        style: 'tooltipOnly';\n      }\n  );\n\n  export type Stack = Series[];\n}\n","<div\n  *ngIf=\"!(empty$ | async); else zeroState\"\n  class=\"container\"\n  (rivClientSize)=\"width = $event.width\"\n>\n  <svg\n    xmlns=\"http://www.w3.org/2000/svg\"\n    *ngIf=\"drawData$ | async; let d\"\n    [attr.viewBox]=\"d.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"d.columnWidth\"\n        [attr.height]=\"d.columnWidth\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          x1=\"0\"\n          [attr.y1]=\"d.columnWidth\"\n          [attr.x2]=\"d.columnWidth\"\n          y2=\"0\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let tick of d.xMinorTicks\">\n      <rect\n        class=\"tick-background\"\n        [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n        [attr.x]=\"tick.x - 20\"\n        [attr.y]=\"d.yScale(0) + 3\"\n        width=\"40\"\n        height=\"12\"\n        rx=\"2\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 12\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.xMajorTicks\">\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 24\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.yTicks\">\n      <rect\n        class=\"tick\"\n        x=\"0\"\n        [attr.y]=\"d.yScale(tick)\"\n        width=\"100%\"\n        height=\"1\"\n      ></rect>\n      <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g *ngFor=\"let rect of d.rects\">\n      <rect\n        class=\"column\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        [attr.fill]=\"rect.fill\"\n        [class.blurred]=\"\n          (hoveredBand$ | async) !== null &&\n          (hoveredBand$ | async)?.dateValue !== rect.dateValue\n        \"\n      ></rect>\n      <rect\n        *ngIf=\"rect.striped\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        fill=\"url(#stripes)\"\n      ></rect>\n    </g>\n    <rect\n      *ngFor=\"let rect of d.hoverBands\"\n      [attr.x]=\"rect.x\"\n      [attr.y]=\"rect.y\"\n      [attr.width]=\"rect.width\"\n      [attr.height]=\"rect.height\"\n      fill=\"transparent\"\n      (mouseenter)=\"\n        hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n      \"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    ></rect>\n  </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n  <riv-callout\n    *riv-callout\n    [anchor]=\"callout.anchor\"\n    [isModal]=\"false\"\n    [preferredPosition]=\"'center-right'\"\n    [allowedPositions]=\"[\n      'center-right',\n      'center-left',\n      'top-center',\n      'bottom-center'\n    ]\"\n  >\n    <div class=\"callout-content\">\n      <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n        <div>{{ metric.label }}</div>\n        <div class=\"callout-metric-value\">{{ metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state></riv-zero-state>\n</ng-template>\n\n<ng-template #controls>\n  <riv-single-select\n    [groups]=\"intervalOptions$ | async | rivOptionGroup\"\n    [selectedOption]=\"selectedIntervalOption$ | async\"\n    (selectedOptionChange)=\"setIntervalOption($event)\"\n  ></riv-single-select>\n</ng-template>\n"]}
@@ -15,13 +15,4 @@ export function getDateRange(stacks) {
15
15
  export function pointReducer(points) {
16
16
  return points.reduce((total, { value }) => total + value, 0);
17
17
  }
18
- export function getColumnWidth(numColumns) {
19
- if (numColumns <= 6)
20
- return 72;
21
- if (numColumns <= 18)
22
- return 40;
23
- if (numColumns <= 40)
24
- return 16;
25
- return 8;
26
- }
27
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2tlZC1jb2x1bW4uaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jpdi9zcmMvbGliL3Zpc3VhbGl6YXRpb24vc3RhY2tlZC1jb2x1bW4vc3RhY2tlZC1jb2x1bW4uaGVscGVycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxNQUFNLFVBQVUsWUFBWSxDQUMxQixNQUFzQztJQUV0QyxJQUFJLEdBQUcsR0FBRyxRQUFRLENBQUM7SUFDbkIsSUFBSSxHQUFHLEdBQUcsQ0FBQyxRQUFRLENBQUM7SUFDcEIsS0FBSyxNQUFNLEtBQUssSUFBSSxNQUFNLEVBQUU7UUFDMUIsS0FBSyxNQUFNLE1BQU0sSUFBSSxLQUFLLEVBQUU7WUFDMUIsS0FBSyxNQUFNLEVBQUUsSUFBSSxFQUFFLElBQUksTUFBTSxDQUFDLElBQUksRUFBRTtnQkFDbEMsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO2dCQUM3QixHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUM7Z0JBQzNCLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsQ0FBQzthQUM1QjtTQUNGO0tBQ0Y7SUFDRCxPQUFPLENBQUMsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUN4QyxDQUFDO0FBRUQsTUFBTSxVQUFVLFlBQVksQ0FBQyxNQUFzQztJQUNqRSxPQUFPLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFLENBQUMsS0FBSyxHQUFHLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FBQztBQUMvRCxDQUFDO0FBRUQsTUFBTSxVQUFVLGNBQWMsQ0FBQyxVQUFrQjtJQUMvQyxJQUFJLFVBQVUsSUFBSSxDQUFDO1FBQUUsT0FBTyxFQUFFLENBQUM7SUFDL0IsSUFBSSxVQUFVLElBQUksRUFBRTtRQUFFLE9BQU8sRUFBRSxDQUFDO0lBQ2hDLElBQUksVUFBVSxJQUFJLEVBQUU7UUFBRSxPQUFPLEVBQUUsQ0FBQztJQUNoQyxPQUFPLENBQUMsQ0FBQztBQUNYLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTdGFja2VkQ29sdW1uQ29tcG9uZW50IH0gZnJvbSAnLi9zdGFja2VkLWNvbHVtbi5jb21wb25lbnQnO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0RGF0ZVJhbmdlKFxuICBzdGFja3M6IFN0YWNrZWRDb2x1bW5Db21wb25lbnQuU3RhY2tbXSxcbik6IFtEYXRlLCBEYXRlXSB7XG4gIGxldCBtaW4gPSBJbmZpbml0eTtcbiAgbGV0IG1heCA9IC1JbmZpbml0eTtcbiAgZm9yIChjb25zdCBzdGFjayBvZiBzdGFja3MpIHtcbiAgICBmb3IgKGNvbnN0IHNlcmllcyBvZiBzdGFjaykge1xuICAgICAgZm9yIChjb25zdCB7IGRhdGUgfSBvZiBzZXJpZXMuZGF0YSkge1xuICAgICAgICBjb25zdCB2YWx1ZSA9IGRhdGUudmFsdWVPZigpO1xuICAgICAgICBtaW4gPSBNYXRoLm1pbihtaW4sIHZhbHVlKTtcbiAgICAgICAgbWF4ID0gTWF0aC5tYXgobWF4LCB2YWx1ZSk7XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHJldHVybiBbbmV3IERhdGUobWluKSwgbmV3IERhdGUobWF4KV07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwb2ludFJlZHVjZXIocG9pbnRzOiBTdGFja2VkQ29sdW1uQ29tcG9uZW50LlZhbHVlW10pIHtcbiAgcmV0dXJuIHBvaW50cy5yZWR1Y2UoKHRvdGFsLCB7IHZhbHVlIH0pID0+IHRvdGFsICsgdmFsdWUsIDApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0Q29sdW1uV2lkdGgobnVtQ29sdW1uczogbnVtYmVyKSB7XG4gIGlmIChudW1Db2x1bW5zIDw9IDYpIHJldHVybiA3MjtcbiAgaWYgKG51bUNvbHVtbnMgPD0gMTgpIHJldHVybiA0MDtcbiAgaWYgKG51bUNvbHVtbnMgPD0gNDApIHJldHVybiAxNjtcbiAgcmV0dXJuIDg7XG59XG4iXX0=
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2tlZC1jb2x1bW4uaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jpdi9zcmMvbGliL3Zpc3VhbGl6YXRpb24vc3RhY2tlZC1jb2x1bW4vc3RhY2tlZC1jb2x1bW4uaGVscGVycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxNQUFNLFVBQVUsWUFBWSxDQUMxQixNQUFzQztJQUV0QyxJQUFJLEdBQUcsR0FBRyxRQUFRLENBQUM7SUFDbkIsSUFBSSxHQUFHLEdBQUcsQ0FBQyxRQUFRLENBQUM7SUFDcEIsS0FBSyxNQUFNLEtBQUssSUFBSSxNQUFNLEVBQUU7UUFDMUIsS0FBSyxNQUFNLE1BQU0sSUFBSSxLQUFLLEVBQUU7WUFDMUIsS0FBSyxNQUFNLEVBQUUsSUFBSSxFQUFFLElBQUksTUFBTSxDQUFDLElBQUksRUFBRTtnQkFDbEMsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO2dCQUM3QixHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUM7Z0JBQzNCLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsQ0FBQzthQUM1QjtTQUNGO0tBQ0Y7SUFDRCxPQUFPLENBQUMsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUN4QyxDQUFDO0FBRUQsTUFBTSxVQUFVLFlBQVksQ0FBQyxNQUFzQztJQUNqRSxPQUFPLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFLENBQUMsS0FBSyxHQUFHLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FBQztBQUMvRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU3RhY2tlZENvbHVtbkNvbXBvbmVudCB9IGZyb20gJy4vc3RhY2tlZC1jb2x1bW4uY29tcG9uZW50JztcblxuZXhwb3J0IGZ1bmN0aW9uIGdldERhdGVSYW5nZShcbiAgc3RhY2tzOiBTdGFja2VkQ29sdW1uQ29tcG9uZW50LlN0YWNrW10sXG4pOiBbRGF0ZSwgRGF0ZV0ge1xuICBsZXQgbWluID0gSW5maW5pdHk7XG4gIGxldCBtYXggPSAtSW5maW5pdHk7XG4gIGZvciAoY29uc3Qgc3RhY2sgb2Ygc3RhY2tzKSB7XG4gICAgZm9yIChjb25zdCBzZXJpZXMgb2Ygc3RhY2spIHtcbiAgICAgIGZvciAoY29uc3QgeyBkYXRlIH0gb2Ygc2VyaWVzLmRhdGEpIHtcbiAgICAgICAgY29uc3QgdmFsdWUgPSBkYXRlLnZhbHVlT2YoKTtcbiAgICAgICAgbWluID0gTWF0aC5taW4obWluLCB2YWx1ZSk7XG4gICAgICAgIG1heCA9IE1hdGgubWF4KG1heCwgdmFsdWUpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuICByZXR1cm4gW25ldyBEYXRlKG1pbiksIG5ldyBEYXRlKG1heCldO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcG9pbnRSZWR1Y2VyKHBvaW50czogU3RhY2tlZENvbHVtbkNvbXBvbmVudC5WYWx1ZVtdKSB7XG4gIHJldHVybiBwb2ludHMucmVkdWNlKCh0b3RhbCwgeyB2YWx1ZSB9KSA9PiB0b3RhbCArIHZhbHVlLCAwKTtcbn1cbiJdfQ==