@rivet-health/design-system 5.1.2 → 5.2.1

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.
@@ -9,17 +9,21 @@ export class LegendItemComponent {
9
9
  this.colorToken = '';
10
10
  this.style = 'solid';
11
11
  this.visibility = 'visible';
12
+ this.clickable = false;
12
13
  this.itemClick = new EventEmitter();
13
14
  }
14
15
  get backgroundColor() {
15
16
  return `var(${this.colorToken})`;
16
17
  }
18
+ isLegendItemClickable() {
19
+ return this.clickable && this.visibility !== 'locked';
20
+ }
17
21
  }
18
22
  LegendItemComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: LegendItemComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
19
- LegendItemComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: LegendItemComponent, selector: "riv-legend-item", inputs: { label: "label", colorToken: "colorToken", style: "style", visibility: "visibility", iconTooltip: "iconTooltip" }, outputs: { itemClick: "itemClick" }, ngImport: i0, template: "<button (click)=\"itemClick.emit()\" [rivTooltip]=\"iconTooltip\" type=\"button\">\n <i\n *ngIf=\"visibility === 'visible'\"\n [class.striped]=\"style === 'striped'\"\n [style.background-color]=\"backgroundColor\"\n ></i>\n <riv-icon\n *ngIf=\"visibility === 'hidden'\"\n name=\"EyeOff\"\n [size]=\"16\"\n ></riv-icon>\n <riv-icon *ngIf=\"visibility === 'locked'\" name=\"Lock\" [size]=\"16\"></riv-icon>\n</button>\n<span [class.low-contrast]=\"visibility === 'hidden' || visibility === 'locked'\">\n {{ label }}\n</span>\n", styles: [":host{display:inline-flex;align-items:center;gap:var(--size-small)}i{width:var(--size-large);height:var(--size-large)}i.striped{background-image:repeating-linear-gradient(315deg,transparent 0px,transparent 2.5px,var(--white-100) 2.5px,var(--white-100) 3.5px,transparent 3.5px,transparent 6px)}span{color:var(--type-light-low-contrast);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}button{display:inline-flex}riv-icon,span.low-contrast{color:var(--type-light-disabled)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "directive", type: i3.TooltipDirective, selector: "[rivTooltip]", inputs: ["rivTooltip", "rivTooltipTheme", "rivTooltipMaxWidth", "rivTooltipPreferredPosition", "rivTooltipCloseDelay"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
23
+ LegendItemComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: LegendItemComponent, selector: "riv-legend-item", inputs: { label: "label", colorToken: "colorToken", style: "style", visibility: "visibility", iconTooltip: "iconTooltip", clickable: "clickable" }, outputs: { itemClick: "itemClick" }, ngImport: i0, template: "<button\n (click)=\"isLegendItemClickable() ? itemClick.emit() : null\"\n [rivTooltip]=\"iconTooltip\"\n type=\"button\"\n [disabled]=\"visibility === 'locked'\"\n [class.clickable]=\"clickable\"\n>\n <i\n *ngIf=\"visibility === 'visible'\"\n [class.striped]=\"style === 'striped'\"\n [style.background-color]=\"backgroundColor\"\n class=\"opacity-hover\"\n ></i>\n <i\n *ngIf=\"visibility === 'hidden'\"\n class=\"hidden-legend-item opacity-hover\"\n [class.striped]=\"style === 'striped'\"\n [style.border-color]=\"backgroundColor\"\n [style.--background-color-hover]=\"backgroundColor\"\n ></i>\n <riv-icon *ngIf=\"visibility === 'locked'\" name=\"Lock\" [size]=\"16\"></riv-icon>\n\n <span\n [class.low-contrast]=\"visibility === 'hidden' || visibility === 'locked'\"\n >\n {{ label }}\n </span>\n</button>\n", styles: [":host{display:inline-flex;align-items:center;gap:var(--size-small)}i{width:var(--size-large);height:var(--size-large)}i.striped{background-image:repeating-linear-gradient(315deg,transparent 0px,transparent 2.5px,var(--white-100) 2.5px,var(--white-100) 3.5px,transparent 3.5px,transparent 6px)}span{color:var(--type-light-low-contrast);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}button{display:inline-flex;gap:var(--size-small);align-items:center;cursor:default}.hidden-legend-item{border-width:var(--border-width);border-style:solid;background-color:transparent}button:hover.clickable .hidden-legend-item{background-color:var(--background-color-hover)}button:not([disabled]).clickable:hover{cursor:pointer}button:not([disabled]).clickable:hover .opacity-hover{opacity:.6}button[disabled]{cursor:not-allowed}button[disabled] .low-contrast{color:var(--type-light-disabled)}button:not(:hover) .low-contrast{color:var(--type-light-disabled)}button:not(.clickable):hover .low-contrast{color:var(--type-light-disabled)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "directive", type: i3.TooltipDirective, selector: "[rivTooltip]", inputs: ["rivTooltip", "rivTooltipTheme", "rivTooltipMaxWidth", "rivTooltipPreferredPosition", "rivTooltipCloseDelay"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
20
24
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: LegendItemComponent, decorators: [{
21
25
  type: Component,
22
- args: [{ selector: 'riv-legend-item', changeDetection: ChangeDetectionStrategy.OnPush, template: "<button (click)=\"itemClick.emit()\" [rivTooltip]=\"iconTooltip\" type=\"button\">\n <i\n *ngIf=\"visibility === 'visible'\"\n [class.striped]=\"style === 'striped'\"\n [style.background-color]=\"backgroundColor\"\n ></i>\n <riv-icon\n *ngIf=\"visibility === 'hidden'\"\n name=\"EyeOff\"\n [size]=\"16\"\n ></riv-icon>\n <riv-icon *ngIf=\"visibility === 'locked'\" name=\"Lock\" [size]=\"16\"></riv-icon>\n</button>\n<span [class.low-contrast]=\"visibility === 'hidden' || visibility === 'locked'\">\n {{ label }}\n</span>\n", styles: [":host{display:inline-flex;align-items:center;gap:var(--size-small)}i{width:var(--size-large);height:var(--size-large)}i.striped{background-image:repeating-linear-gradient(315deg,transparent 0px,transparent 2.5px,var(--white-100) 2.5px,var(--white-100) 3.5px,transparent 3.5px,transparent 6px)}span{color:var(--type-light-low-contrast);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}button{display:inline-flex}riv-icon,span.low-contrast{color:var(--type-light-disabled)}\n"] }]
26
+ args: [{ selector: 'riv-legend-item', changeDetection: ChangeDetectionStrategy.OnPush, template: "<button\n (click)=\"isLegendItemClickable() ? itemClick.emit() : null\"\n [rivTooltip]=\"iconTooltip\"\n type=\"button\"\n [disabled]=\"visibility === 'locked'\"\n [class.clickable]=\"clickable\"\n>\n <i\n *ngIf=\"visibility === 'visible'\"\n [class.striped]=\"style === 'striped'\"\n [style.background-color]=\"backgroundColor\"\n class=\"opacity-hover\"\n ></i>\n <i\n *ngIf=\"visibility === 'hidden'\"\n class=\"hidden-legend-item opacity-hover\"\n [class.striped]=\"style === 'striped'\"\n [style.border-color]=\"backgroundColor\"\n [style.--background-color-hover]=\"backgroundColor\"\n ></i>\n <riv-icon *ngIf=\"visibility === 'locked'\" name=\"Lock\" [size]=\"16\"></riv-icon>\n\n <span\n [class.low-contrast]=\"visibility === 'hidden' || visibility === 'locked'\"\n >\n {{ label }}\n </span>\n</button>\n", styles: [":host{display:inline-flex;align-items:center;gap:var(--size-small)}i{width:var(--size-large);height:var(--size-large)}i.striped{background-image:repeating-linear-gradient(315deg,transparent 0px,transparent 2.5px,var(--white-100) 2.5px,var(--white-100) 3.5px,transparent 3.5px,transparent 6px)}span{color:var(--type-light-low-contrast);font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0)}button{display:inline-flex;gap:var(--size-small);align-items:center;cursor:default}.hidden-legend-item{border-width:var(--border-width);border-style:solid;background-color:transparent}button:hover.clickable .hidden-legend-item{background-color:var(--background-color-hover)}button:not([disabled]).clickable:hover{cursor:pointer}button:not([disabled]).clickable:hover .opacity-hover{opacity:.6}button[disabled]{cursor:not-allowed}button[disabled] .low-contrast{color:var(--type-light-disabled)}button:not(:hover) .low-contrast{color:var(--type-light-disabled)}button:not(.clickable):hover .low-contrast{color:var(--type-light-disabled)}\n"] }]
23
27
  }], propDecorators: { label: [{
24
28
  type: Input
25
29
  }], colorToken: [{
@@ -30,6 +34,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
30
34
  type: Input
31
35
  }], iconTooltip: [{
32
36
  type: Input
37
+ }], clickable: [{
38
+ type: Input
33
39
  }], itemClick: [{
34
40
  type: Output
35
41
  }] } });
@@ -37,4 +43,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
37
43
  LegendItemComponent.Styles = ['solid', 'striped'];
38
44
  LegendItemComponent.Visibilities = ['visible', 'hidden', 'locked'];
39
45
  })(LegendItemComponent || (LegendItemComponent = {}));
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGVnZW5kLWl0ZW0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvdmlzdWFsaXphdGlvbi9sZWdlbmQtaXRlbS9sZWdlbmQtaXRlbS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi92aXN1YWxpemF0aW9uL2xlZ2VuZC1pdGVtL2xlZ2VuZC1pdGVtLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCx1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFlBQVksRUFDWixLQUFLLEVBQ0wsTUFBTSxHQUNQLE1BQU0sZUFBZSxDQUFDOzs7OztBQVF2QixNQUFNLE9BQU8sbUJBQW1CO0lBTmhDO1FBUUUsVUFBSyxHQUFXLEVBQUUsQ0FBQztRQUduQixlQUFVLEdBQVcsRUFBRSxDQUFDO1FBR3hCLFVBQUssR0FBOEIsT0FBTyxDQUFDO1FBRzNDLGVBQVUsR0FBbUMsU0FBUyxDQUFDO1FBVXZELGNBQVMsR0FBRyxJQUFJLFlBQVksRUFBRSxDQUFDO0tBQ2hDO0lBTkMsSUFBSSxlQUFlO1FBQ2pCLE9BQU8sT0FBTyxJQUFJLENBQUMsVUFBVSxHQUFHLENBQUM7SUFDbkMsQ0FBQzs7Z0hBbEJVLG1CQUFtQjtvR0FBbkIsbUJBQW1CLHdOQ2RoQyxzaUJBZ0JBOzJGREZhLG1CQUFtQjtrQkFOL0IsU0FBUzsrQkFDRSxpQkFBaUIsbUJBR1YsdUJBQXVCLENBQUMsTUFBTTs4QkFJL0MsS0FBSztzQkFESixLQUFLO2dCQUlOLFVBQVU7c0JBRFQsS0FBSztnQkFJTixLQUFLO3NCQURKLEtBQUs7Z0JBSU4sVUFBVTtzQkFEVCxLQUFLO2dCQUlOLFdBQVc7c0JBRFYsS0FBSztnQkFRTixTQUFTO3NCQURSLE1BQU07O0FBSVQsV0FBaUIsbUJBQW1CO0lBQ3JCLDBCQUFNLEdBQUcsQ0FBQyxPQUFPLEVBQUUsU0FBUyxDQUFVLENBQUM7SUFHdkMsZ0NBQVksR0FBRyxDQUFDLFNBQVMsRUFBRSxRQUFRLEVBQUUsUUFBUSxDQUFVLENBQUM7QUFFdkUsQ0FBQyxFQU5nQixtQkFBbUIsS0FBbkIsbUJBQW1CLFFBTW5DIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG4gIENvbXBvbmVudCxcbiAgRXZlbnRFbWl0dGVyLFxuICBJbnB1dCxcbiAgT3V0cHV0LFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAncml2LWxlZ2VuZC1pdGVtJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2xlZ2VuZC1pdGVtLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbGVnZW5kLWl0ZW0uY29tcG9uZW50LmNzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgTGVnZW5kSXRlbUNvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIGxhYmVsOiBzdHJpbmcgPSAnJztcblxuICBASW5wdXQoKVxuICBjb2xvclRva2VuOiBzdHJpbmcgPSAnJztcblxuICBASW5wdXQoKVxuICBzdHlsZTogTGVnZW5kSXRlbUNvbXBvbmVudC5TdHlsZSA9ICdzb2xpZCc7XG5cbiAgQElucHV0KClcbiAgdmlzaWJpbGl0eTogTGVnZW5kSXRlbUNvbXBvbmVudC5WaXNpYmlsaXR5ID0gJ3Zpc2libGUnO1xuXG4gIEBJbnB1dCgpXG4gIGljb25Ub29sdGlwPzogc3RyaW5nO1xuXG4gIGdldCBiYWNrZ3JvdW5kQ29sb3IoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYHZhcigke3RoaXMuY29sb3JUb2tlbn0pYDtcbiAgfVxuXG4gIEBPdXRwdXQoKVxuICBpdGVtQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyKCk7XG59XG5cbmV4cG9ydCBuYW1lc3BhY2UgTGVnZW5kSXRlbUNvbXBvbmVudCB7XG4gIGV4cG9ydCBjb25zdCBTdHlsZXMgPSBbJ3NvbGlkJywgJ3N0cmlwZWQnXSBhcyBjb25zdDtcbiAgZXhwb3J0IHR5cGUgU3R5bGUgPSAodHlwZW9mIFN0eWxlcylbbnVtYmVyXTtcblxuICBleHBvcnQgY29uc3QgVmlzaWJpbGl0aWVzID0gWyd2aXNpYmxlJywgJ2hpZGRlbicsICdsb2NrZWQnXSBhcyBjb25zdDtcbiAgZXhwb3J0IHR5cGUgVmlzaWJpbGl0eSA9ICh0eXBlb2YgVmlzaWJpbGl0aWVzKVtudW1iZXJdO1xufVxuIiwiPGJ1dHRvbiAoY2xpY2spPVwiaXRlbUNsaWNrLmVtaXQoKVwiIFtyaXZUb29sdGlwXT1cImljb25Ub29sdGlwXCIgdHlwZT1cImJ1dHRvblwiPlxuICA8aVxuICAgICpuZ0lmPVwidmlzaWJpbGl0eSA9PT0gJ3Zpc2libGUnXCJcbiAgICBbY2xhc3Muc3RyaXBlZF09XCJzdHlsZSA9PT0gJ3N0cmlwZWQnXCJcbiAgICBbc3R5bGUuYmFja2dyb3VuZC1jb2xvcl09XCJiYWNrZ3JvdW5kQ29sb3JcIlxuICA+PC9pPlxuICA8cml2LWljb25cbiAgICAqbmdJZj1cInZpc2liaWxpdHkgPT09ICdoaWRkZW4nXCJcbiAgICBuYW1lPVwiRXllT2ZmXCJcbiAgICBbc2l6ZV09XCIxNlwiXG4gID48L3Jpdi1pY29uPlxuICA8cml2LWljb24gKm5nSWY9XCJ2aXNpYmlsaXR5ID09PSAnbG9ja2VkJ1wiIG5hbWU9XCJMb2NrXCIgW3NpemVdPVwiMTZcIj48L3Jpdi1pY29uPlxuPC9idXR0b24+XG48c3BhbiBbY2xhc3MubG93LWNvbnRyYXN0XT1cInZpc2liaWxpdHkgPT09ICdoaWRkZW4nIHx8IHZpc2liaWxpdHkgPT09ICdsb2NrZWQnXCI+XG4gIHt7IGxhYmVsIH19XG48L3NwYW4+XG4iXX0=
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGVnZW5kLWl0ZW0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvdmlzdWFsaXphdGlvbi9sZWdlbmQtaXRlbS9sZWdlbmQtaXRlbS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi92aXN1YWxpemF0aW9uL2xlZ2VuZC1pdGVtL2xlZ2VuZC1pdGVtLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCx1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFlBQVksRUFDWixLQUFLLEVBQ0wsTUFBTSxHQUNQLE1BQU0sZUFBZSxDQUFDOzs7OztBQVF2QixNQUFNLE9BQU8sbUJBQW1CO0lBTmhDO1FBUUUsVUFBSyxHQUFXLEVBQUUsQ0FBQztRQUduQixlQUFVLEdBQVcsRUFBRSxDQUFDO1FBR3hCLFVBQUssR0FBOEIsT0FBTyxDQUFDO1FBRzNDLGVBQVUsR0FBbUMsU0FBUyxDQUFDO1FBVXZELGNBQVMsR0FBWSxLQUFLLENBQUM7UUFHM0IsY0FBUyxHQUFHLElBQUksWUFBWSxFQUFFLENBQUM7S0FLaEM7SUFiQyxJQUFJLGVBQWU7UUFDakIsT0FBTyxPQUFPLElBQUksQ0FBQyxVQUFVLEdBQUcsQ0FBQztJQUNuQyxDQUFDO0lBUUQscUJBQXFCO1FBQ25CLE9BQU8sSUFBSSxDQUFDLFNBQVMsSUFBSSxJQUFJLENBQUMsVUFBVSxLQUFLLFFBQVEsQ0FBQztJQUN4RCxDQUFDOztnSEE1QlUsbUJBQW1CO29HQUFuQixtQkFBbUIsZ1BDZGhDLDQxQkE0QkE7MkZEZGEsbUJBQW1CO2tCQU4vQixTQUFTOytCQUNFLGlCQUFpQixtQkFHVix1QkFBdUIsQ0FBQyxNQUFNOzhCQUkvQyxLQUFLO3NCQURKLEtBQUs7Z0JBSU4sVUFBVTtzQkFEVCxLQUFLO2dCQUlOLEtBQUs7c0JBREosS0FBSztnQkFJTixVQUFVO3NCQURULEtBQUs7Z0JBSU4sV0FBVztzQkFEVixLQUFLO2dCQVFOLFNBQVM7c0JBRFIsS0FBSztnQkFJTixTQUFTO3NCQURSLE1BQU07O0FBUVQsV0FBaUIsbUJBQW1CO0lBQ3JCLDBCQUFNLEdBQUcsQ0FBQyxPQUFPLEVBQUUsU0FBUyxDQUFVLENBQUM7SUFHdkMsZ0NBQVksR0FBRyxDQUFDLFNBQVMsRUFBRSxRQUFRLEVBQUUsUUFBUSxDQUFVLENBQUM7QUFFdkUsQ0FBQyxFQU5nQixtQkFBbUIsS0FBbkIsbUJBQW1CLFFBTW5DIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG4gIENvbXBvbmVudCxcbiAgRXZlbnRFbWl0dGVyLFxuICBJbnB1dCxcbiAgT3V0cHV0LFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAncml2LWxlZ2VuZC1pdGVtJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2xlZ2VuZC1pdGVtLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbGVnZW5kLWl0ZW0uY29tcG9uZW50LmNzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgTGVnZW5kSXRlbUNvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIGxhYmVsOiBzdHJpbmcgPSAnJztcblxuICBASW5wdXQoKVxuICBjb2xvclRva2VuOiBzdHJpbmcgPSAnJztcblxuICBASW5wdXQoKVxuICBzdHlsZTogTGVnZW5kSXRlbUNvbXBvbmVudC5TdHlsZSA9ICdzb2xpZCc7XG5cbiAgQElucHV0KClcbiAgdmlzaWJpbGl0eTogTGVnZW5kSXRlbUNvbXBvbmVudC5WaXNpYmlsaXR5ID0gJ3Zpc2libGUnO1xuXG4gIEBJbnB1dCgpXG4gIGljb25Ub29sdGlwPzogc3RyaW5nO1xuXG4gIGdldCBiYWNrZ3JvdW5kQ29sb3IoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gYHZhcigke3RoaXMuY29sb3JUb2tlbn0pYDtcbiAgfVxuXG4gIEBJbnB1dCgpXG4gIGNsaWNrYWJsZTogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIEBPdXRwdXQoKVxuICBpdGVtQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyKCk7XG5cbiAgaXNMZWdlbmRJdGVtQ2xpY2thYmxlKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLmNsaWNrYWJsZSAmJiB0aGlzLnZpc2liaWxpdHkgIT09ICdsb2NrZWQnO1xuICB9XG59XG5cbmV4cG9ydCBuYW1lc3BhY2UgTGVnZW5kSXRlbUNvbXBvbmVudCB7XG4gIGV4cG9ydCBjb25zdCBTdHlsZXMgPSBbJ3NvbGlkJywgJ3N0cmlwZWQnXSBhcyBjb25zdDtcbiAgZXhwb3J0IHR5cGUgU3R5bGUgPSAodHlwZW9mIFN0eWxlcylbbnVtYmVyXTtcblxuICBleHBvcnQgY29uc3QgVmlzaWJpbGl0aWVzID0gWyd2aXNpYmxlJywgJ2hpZGRlbicsICdsb2NrZWQnXSBhcyBjb25zdDtcbiAgZXhwb3J0IHR5cGUgVmlzaWJpbGl0eSA9ICh0eXBlb2YgVmlzaWJpbGl0aWVzKVtudW1iZXJdO1xufVxuIiwiPGJ1dHRvblxuICAoY2xpY2spPVwiaXNMZWdlbmRJdGVtQ2xpY2thYmxlKCkgPyBpdGVtQ2xpY2suZW1pdCgpIDogbnVsbFwiXG4gIFtyaXZUb29sdGlwXT1cImljb25Ub29sdGlwXCJcbiAgdHlwZT1cImJ1dHRvblwiXG4gIFtkaXNhYmxlZF09XCJ2aXNpYmlsaXR5ID09PSAnbG9ja2VkJ1wiXG4gIFtjbGFzcy5jbGlja2FibGVdPVwiY2xpY2thYmxlXCJcbj5cbiAgPGlcbiAgICAqbmdJZj1cInZpc2liaWxpdHkgPT09ICd2aXNpYmxlJ1wiXG4gICAgW2NsYXNzLnN0cmlwZWRdPVwic3R5bGUgPT09ICdzdHJpcGVkJ1wiXG4gICAgW3N0eWxlLmJhY2tncm91bmQtY29sb3JdPVwiYmFja2dyb3VuZENvbG9yXCJcbiAgICBjbGFzcz1cIm9wYWNpdHktaG92ZXJcIlxuICA+PC9pPlxuICA8aVxuICAgICpuZ0lmPVwidmlzaWJpbGl0eSA9PT0gJ2hpZGRlbidcIlxuICAgIGNsYXNzPVwiaGlkZGVuLWxlZ2VuZC1pdGVtIG9wYWNpdHktaG92ZXJcIlxuICAgIFtjbGFzcy5zdHJpcGVkXT1cInN0eWxlID09PSAnc3RyaXBlZCdcIlxuICAgIFtzdHlsZS5ib3JkZXItY29sb3JdPVwiYmFja2dyb3VuZENvbG9yXCJcbiAgICBbc3R5bGUuLS1iYWNrZ3JvdW5kLWNvbG9yLWhvdmVyXT1cImJhY2tncm91bmRDb2xvclwiXG4gID48L2k+XG4gIDxyaXYtaWNvbiAqbmdJZj1cInZpc2liaWxpdHkgPT09ICdsb2NrZWQnXCIgbmFtZT1cIkxvY2tcIiBbc2l6ZV09XCIxNlwiPjwvcml2LWljb24+XG5cbiAgPHNwYW5cbiAgICBbY2xhc3MubG93LWNvbnRyYXN0XT1cInZpc2liaWxpdHkgPT09ICdoaWRkZW4nIHx8IHZpc2liaWxpdHkgPT09ICdsb2NrZWQnXCJcbiAgPlxuICAgIHt7IGxhYmVsIH19XG4gIDwvc3Bhbj5cbjwvYnV0dG9uPlxuIl19
@@ -18,7 +18,7 @@ export class StackedRowComponent {
18
18
  // Width helps determine the size of the viewbox this graph is displaying in.
19
19
  this.width$ = new BehaviorSubject(960);
20
20
  // Height helps determine the size of the viewbox this graph is displaying in.
21
- this.height$ = new BehaviorSubject(256);
21
+ this.height$ = new BehaviorSubject(null);
22
22
  // ValueFormatter determines how we style the values in a metric and the x-axis
23
23
  this.valueFormatter = v => v.toString();
24
24
  this.zeroStateMessage$ = new BehaviorSubject('No data to display.');
@@ -28,13 +28,43 @@ export class StackedRowComponent {
28
28
  | X Axis Labels |
29
29
  Row Name | Rectangles | Totals
30
30
  */
31
+ this.RECTANGLE_HEIGHTS = {
32
+ MIN_HEIGHT: 12,
33
+ MAX_HEIGHT: 42, // Sets the largest size a row can expand to
34
+ };
31
35
  this.SECTION_HEIGHTS = {
32
36
  // Space reserved for the x-axis labels
33
37
  X_AXIS_LABEL: 14,
34
38
  // Height of each row
35
- PER_ROW: 20,
36
- // Heigh of each rectangle
37
- PER_RECTANGLE: 8,
39
+ PER_ROW: () => {
40
+ //Only use height if specified
41
+ if (this.input.length && this.height) {
42
+ let rowHeight = (this.height -
43
+ this.SECTION_HEIGHTS.X_AXIS_LABEL -
44
+ this.SECTION_PADDING.OUTER_TOP_BOTTOM -
45
+ this.SECTION_PADDING.RECTANGLE_TO_ROW) /
46
+ this.input.length -
47
+ this.SECTION_PADDING.BETWEEN_ROW; // calculate base row height
48
+ // Clamping the row height between the min and max heights
49
+ return Math.min(Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
50
+ }
51
+ return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input
52
+ },
53
+ // Max Height of each rectangle
54
+ PER_RECTANGLE: () => {
55
+ //Only use height if specified
56
+ if (this.input.length && this.height) {
57
+ let rectangleHeight = (this.height -
58
+ this.SECTION_HEIGHTS.X_AXIS_LABEL -
59
+ this.SECTION_PADDING.OUTER_TOP_BOTTOM -
60
+ this.SECTION_PADDING.RECTANGLE_TO_ROW) /
61
+ this.input.length -
62
+ this.SECTION_PADDING.BETWEEN_ROW; // calculate base rectangle height
63
+ // Clamping the rectangle height between the min and max heights
64
+ return Math.min(Math.max(rectangleHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
65
+ }
66
+ return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input
67
+ },
38
68
  };
39
69
  this.SECTION_WIDTHS = {
40
70
  // Space reserved for the name at the start of the row
@@ -52,33 +82,45 @@ export class StackedRowComponent {
52
82
  RECTANGLE_TO_ROW: 6,
53
83
  };
54
84
  // The number of X-axis segments to create, with labels at the start of each segment.
55
- this.X_TICK_COUNT = 8;
85
+ this.X_TICK_COUNT_MAX = 8;
86
+ this.X_TICK_COUNT_MIN = 3;
87
+ this.X_TICK_MAX_WIDTH = 900;
88
+ this.X_TICK_MIN_WIDTH = 400;
56
89
  this.drawData$ = combineLatest([
57
90
  this.input$,
58
91
  this.width$,
59
92
  this.height$,
60
93
  ]).pipe(map(([input, width, height]) => {
61
- return this.getDrawnData(input, width, height);
94
+ return this.getDrawnData(input, width, height
95
+ ? height
96
+ : this.SECTION_HEIGHTS.X_AXIS_LABEL +
97
+ this.SECTION_PADDING.OUTER_TOP_BOTTOM +
98
+ this.SECTION_PADDING.RECTANGLE_TO_ROW +
99
+ (this.SECTION_HEIGHTS.PER_ROW() +
100
+ this.SECTION_PADDING.BETWEEN_ROW) *
101
+ input.length);
62
102
  }), shareReplay({ refCount: true, bufferSize: 1 }));
63
103
  // Lets us know whether to display the zero data state
64
104
  this.empty$ = this.input$.pipe(map(input => input.every(stack => stack.data.length === 0)), shareReplay({ refCount: true, bufferSize: 1 }));
65
105
  // What rectangle we are currently hovering over
66
106
  this.hoveredRectangle$ = new BehaviorSubject(null);
107
+ // What band we are currently hovering over
108
+ this.hoveredBand$ = new BehaviorSubject(null);
67
109
  // Create the metric to display based on the currently hovered rectangle
68
- this.callout$ = this.hoveredRectangle$.pipe(map(hoveredRectangle => {
69
- if (!hoveredRectangle)
110
+ this.callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(map(([hoveredBand, drawData]) => {
111
+ if (!hoveredBand || !drawData)
70
112
  return null;
71
- const { target } = hoveredRectangle.event;
113
+ const { target } = hoveredBand.event;
72
114
  if (!target)
73
115
  return null;
74
116
  const anchor = target.getBoundingClientRect();
75
- const { rowName, stackedRowElement } = hoveredRectangle.rectangle;
76
- const metric = {
77
- rowName: rowName,
78
- elementLabel: stackedRowElement.label,
79
- value: this.valueFormatter(stackedRowElement.value),
80
- };
81
- return { anchor, metric };
117
+ const metrics = drawData.rects
118
+ .filter(rect => rect.rowName === hoveredBand.rowName)
119
+ .map(rect => ({
120
+ label: rect.stackedRowElement.label,
121
+ value: this.valueFormatter(rect.stackedRowElement.value),
122
+ }));
123
+ return { anchor, metrics };
82
124
  }), shareReplay({ refCount: true, bufferSize: 1 }));
83
125
  }
84
126
  set input(v) {
@@ -122,7 +164,7 @@ export class StackedRowComponent {
122
164
  // The y-index of the top of the current row
123
165
  getTopOfRow(rowIndex) {
124
166
  const spaceAboveRows = this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL;
125
- const aboveRows = rowIndex * this.SECTION_HEIGHTS.PER_ROW;
167
+ const aboveRows = rowIndex * this.SECTION_HEIGHTS.PER_ROW();
126
168
  const aboveRowPaddings = rowIndex * this.SECTION_PADDING.BETWEEN_ROW;
127
169
  const allAboveRows = aboveRows + aboveRowPaddings;
128
170
  return spaceAboveRows + allAboveRows;
@@ -143,14 +185,20 @@ export class StackedRowComponent {
143
185
  const xScale = scaleLinear()
144
186
  .domain([0, domainMax])
145
187
  .range([this.locationOfXAxisZero(), this.locationOfXAxisEnd(width)]);
146
- const xTicks = xScale.ticks(this.X_TICK_COUNT);
188
+ // Setup a linear scale for tick count based on width
189
+ const tickScale = scaleLinear()
190
+ .domain([this.X_TICK_MIN_WIDTH, this.X_TICK_MAX_WIDTH])
191
+ .range([this.X_TICK_COUNT_MIN, this.X_TICK_COUNT_MAX])
192
+ .clamp(true); // Ensures the scale does not go beyond the defined range
193
+ const xtickCountAdjusted = Math.round(tickScale(width));
194
+ const xTicks = xScale.ticks(xtickCountAdjusted);
147
195
  const rects = this.createRectanglesFromRows(rowsForDisplay, xScale);
148
196
  const rowNamesToDraw = this.getRowNamesToDraw(rowsForDisplay);
149
197
  const rowTotalsToDraw = this.getRowTotalsToDraw(rowsForDisplay, width);
150
198
  const legendItems = getLegendItems(rowsForDisplay, keysInOrder);
151
199
  return {
152
200
  legendItems: legendItems,
153
- rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE,
201
+ rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE(),
154
202
  rects: rects,
155
203
  rowNames: rowNamesToDraw,
156
204
  rowTotals: rowTotalsToDraw,
@@ -166,7 +214,7 @@ export class StackedRowComponent {
166
214
  currentRow.elementsButStacked.forEach((currentElement, _) => {
167
215
  const topOfRow = this.getTopOfRow(rowIndex);
168
216
  const rectangleWidth = xScale(currentElement.topline) - xScale(currentElement.baseline);
169
- const rectangleHeight = this.SECTION_HEIGHTS.PER_RECTANGLE;
217
+ const rectangleHeight = this.SECTION_HEIGHTS.PER_RECTANGLE();
170
218
  const rectangleX = xScale(currentElement.baseline);
171
219
  const rectangleY = topOfRow + this.SECTION_PADDING.RECTANGLE_TO_ROW;
172
220
  const fill = `var(${currentElement.colorToken})`;
@@ -191,7 +239,7 @@ export class StackedRowComponent {
191
239
  // The row name looks right if its base is at the base of the associated rectangle
192
240
  const rowLabelY = this.getTopOfRow(rowIndex) +
193
241
  this.SECTION_PADDING.RECTANGLE_TO_ROW +
194
- this.SECTION_HEIGHTS.PER_RECTANGLE;
242
+ this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;
195
243
  return {
196
244
  label: row.rowName,
197
245
  x: rowLabelX,
@@ -206,7 +254,7 @@ export class StackedRowComponent {
206
254
  // The row totals looks right if its base is at the base of the associated rectangle
207
255
  const rowLabelY = this.getTopOfRow(rowIndex) +
208
256
  this.SECTION_PADDING.RECTANGLE_TO_ROW +
209
- this.SECTION_HEIGHTS.PER_RECTANGLE;
257
+ this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;
210
258
  return {
211
259
  label: this.valueFormatter(row.rowTotal),
212
260
  x: rowLabelX,
@@ -216,10 +264,10 @@ export class StackedRowComponent {
216
264
  }
217
265
  }
218
266
  StackedRowComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedRowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
219
- StackedRowComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: StackedRowComponent, selector: "riv-stacked-row", inputs: { input: "input", width: "width", height: "height", valueFormatter: "valueFormatter", zeroStateMessage: "zeroStateMessage" }, 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 draw\"\n [attr.viewBox]=\"draw.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"draw.rowHeight\"\n [attr.height]=\"draw.rowHeight\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n [attr.x1]=\"0\"\n [attr.y1]=\"0\"\n [attr.x2]=\"draw.rowHeight\"\n [attr.y2]=\"draw.rowHeight\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n <text\n rivSVGTextTruncate\n [text]=\"rowNameToDraw.label\"\n [width]=\"SECTION_WIDTHS.ROW_NAME\"\n class=\"row-title\"\n [attr.x]=\"rowNameToDraw.x\"\n [attr.y]=\"rowNameToDraw.y\"\n ></text>\n </g>\n <g *ngFor=\"let tick of draw.xTicks\">\n <rect\n class=\"tick\"\n [attr.x]=\"draw.xScale(tick)\"\n y=\"0\"\n width=\"1\"\n height=\"100%\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"draw.xScale(tick)\"\n [attr.y]=\"locationOfXAxisTick()\"\n dx=\"4\"\n dy=\"-4\"\n >\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of draw.rects\">\n <rect\n class=\"row\"\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 (hoveredRectangle$ | async) !== null &&\n (hoveredRectangle$ | async)?.rectangle?.rowName !== rect.rowName\n \"\n (mouseenter)=\"\n hoveredRectangle$.next({ rectangle: rect, event: $event })\n \"\n (mouseleave)=\"hoveredRectangle$.next(null)\"\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 (mouseenter)=\"\n hoveredRectangle$.next({ rectangle: rect, event: $event })\n \"\n (mouseleave)=\"hoveredRectangle$.next(null)\"\n ></rect>\n </g>\n <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n <text\n class=\"row-total\"\n [attr.x]=\"rowTotalToDraw.x\"\n [attr.y]=\"rowTotalToDraw.y\"\n text-anchor=\"end\"\n >\n {{ rowTotalToDraw.label }}\n </text>\n </g>\n </svg>\n <legend *ngIf=\"drawData$ | async; let draw\">\n <riv-legend-item\n *ngFor=\"let item of draw.legendItems\"\n [label]=\"item.label\"\n [colorToken]=\"item.colorToken\"\n [style]=\"item.style ? item.style : 'solid'\"\n ></riv-legend-item>\n </legend>\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-title\">{{ callout.metric.rowName }}</div>\n <div class=\"callout-metric\">\n <div class=\"callout-label\">{{ callout.metric.elementLabel }}</div>\n <div class=\"callout-metric-value\">{{ callout.metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-0-font-size);line-height:var(--type-0-line-height-0);fill:var(--type-light-low-contrast)}.row-title{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.row-total{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2);fill:var(--type-light-low-contrast)}.row{transition:opacity var(--short-transition)}.row.blurred{opacity:.4}legend{display:flex;flex-wrap:wrap;gap:var(--size-large);justify-content:flex-start;align-items:baseline}.callout-content{display:grid;padding:var(--size-medium) var(--size-xlarge);gap:var(--size-xsmall)}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall)}.callout-title{font-size:var(--type-3-font-size);line-height:var(--type-3-line-height-2)}.callout-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.callout-metric-value{font-size:var(--type-4-font-size);line-height:var(--type-4-line-height-1)}\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.LegendItemComponent, selector: "riv-legend-item", inputs: ["label", "colorToken", "style", "visibility", "iconTooltip"], outputs: ["itemClick"] }, { kind: "directive", type: i5.SVGTextTruncateDirective, selector: "svg text[rivSVGTextTruncate]", inputs: ["text", "width"] }, { kind: "component", type: i6.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
267
+ StackedRowComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: StackedRowComponent, selector: "riv-stacked-row", inputs: { input: "input", width: "width", height: "height", valueFormatter: "valueFormatter", zeroStateMessage: "zeroStateMessage" }, 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 draw\"\n [attr.viewBox]=\"draw.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"draw.rowHeight\"\n [attr.height]=\"draw.rowHeight\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n [attr.x1]=\"0\"\n [attr.y1]=\"0\"\n [attr.x2]=\"draw.rowHeight\"\n [attr.y2]=\"draw.rowHeight\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n <text\n rivSVGTextTruncate\n [text]=\"rowNameToDraw.label\"\n [width]=\"SECTION_WIDTHS.ROW_NAME\"\n class=\"row-title\"\n [attr.x]=\"rowNameToDraw.x\"\n [attr.y]=\"rowNameToDraw.y\"\n ></text>\n </g>\n <g *ngFor=\"let tick of draw.xTicks\">\n <rect\n class=\"tick\"\n [attr.x]=\"draw.xScale(tick)\"\n y=\"0\"\n width=\"1\"\n height=\"100%\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"draw.xScale(tick)\"\n [attr.y]=\"locationOfXAxisTick()\"\n dx=\"4\"\n dy=\"-4\"\n >\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g\n *ngFor=\"let rect of draw.rects\"\n (mouseenter)=\"hoveredBand$.next({ rowName: rect.rowName, event: $event })\"\n (mouseleave)=\"hoveredBand$.next(null)\"\n >\n <rect\n class=\"row\"\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)?.rowName !== rect.rowName\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 <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n <text\n class=\"row-total\"\n [attr.x]=\"rowTotalToDraw.x\"\n [attr.y]=\"rowTotalToDraw.y\"\n text-anchor=\"end\"\n >\n {{ rowTotalToDraw.label }}\n </text>\n </g>\n </svg>\n <legend *ngIf=\"drawData$ | async; let draw\">\n <riv-legend-item\n *ngFor=\"let item of draw.legendItems\"\n [label]=\"item.label\"\n [colorToken]=\"item.colorToken\"\n [style]=\"item.style ? item.style : 'solid'\"\n ></riv-legend-item>\n </legend>\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]=\"'bottom-center'\"\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 [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-0-font-size);line-height:var(--type-0-line-height-0);fill:var(--type-light-low-contrast)}.row-title{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.row-total{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2);fill:var(--type-light-low-contrast)}.row{transition:opacity var(--short-transition)}.row.blurred{opacity:.4}legend{display:flex;flex-wrap:wrap;gap:var(--size-large);justify-content:flex-start;align-items:baseline}.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.LegendItemComponent, selector: "riv-legend-item", inputs: ["label", "colorToken", "style", "visibility", "iconTooltip", "clickable"], outputs: ["itemClick"] }, { kind: "directive", type: i5.SVGTextTruncateDirective, selector: "svg text[rivSVGTextTruncate]", inputs: ["text", "width"] }, { kind: "component", type: i6.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
220
268
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedRowComponent, decorators: [{
221
269
  type: Component,
222
- args: [{ selector: 'riv-stacked-row', 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 draw\"\n [attr.viewBox]=\"draw.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"draw.rowHeight\"\n [attr.height]=\"draw.rowHeight\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n [attr.x1]=\"0\"\n [attr.y1]=\"0\"\n [attr.x2]=\"draw.rowHeight\"\n [attr.y2]=\"draw.rowHeight\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n <text\n rivSVGTextTruncate\n [text]=\"rowNameToDraw.label\"\n [width]=\"SECTION_WIDTHS.ROW_NAME\"\n class=\"row-title\"\n [attr.x]=\"rowNameToDraw.x\"\n [attr.y]=\"rowNameToDraw.y\"\n ></text>\n </g>\n <g *ngFor=\"let tick of draw.xTicks\">\n <rect\n class=\"tick\"\n [attr.x]=\"draw.xScale(tick)\"\n y=\"0\"\n width=\"1\"\n height=\"100%\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"draw.xScale(tick)\"\n [attr.y]=\"locationOfXAxisTick()\"\n dx=\"4\"\n dy=\"-4\"\n >\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g *ngFor=\"let rect of draw.rects\">\n <rect\n class=\"row\"\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 (hoveredRectangle$ | async) !== null &&\n (hoveredRectangle$ | async)?.rectangle?.rowName !== rect.rowName\n \"\n (mouseenter)=\"\n hoveredRectangle$.next({ rectangle: rect, event: $event })\n \"\n (mouseleave)=\"hoveredRectangle$.next(null)\"\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 (mouseenter)=\"\n hoveredRectangle$.next({ rectangle: rect, event: $event })\n \"\n (mouseleave)=\"hoveredRectangle$.next(null)\"\n ></rect>\n </g>\n <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n <text\n class=\"row-total\"\n [attr.x]=\"rowTotalToDraw.x\"\n [attr.y]=\"rowTotalToDraw.y\"\n text-anchor=\"end\"\n >\n {{ rowTotalToDraw.label }}\n </text>\n </g>\n </svg>\n <legend *ngIf=\"drawData$ | async; let draw\">\n <riv-legend-item\n *ngFor=\"let item of draw.legendItems\"\n [label]=\"item.label\"\n [colorToken]=\"item.colorToken\"\n [style]=\"item.style ? item.style : 'solid'\"\n ></riv-legend-item>\n </legend>\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-title\">{{ callout.metric.rowName }}</div>\n <div class=\"callout-metric\">\n <div class=\"callout-label\">{{ callout.metric.elementLabel }}</div>\n <div class=\"callout-metric-value\">{{ callout.metric.value }}</div>\n </div>\n </div>\n </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n <riv-zero-state [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-0-font-size);line-height:var(--type-0-line-height-0);fill:var(--type-light-low-contrast)}.row-title{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.row-total{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2);fill:var(--type-light-low-contrast)}.row{transition:opacity var(--short-transition)}.row.blurred{opacity:.4}legend{display:flex;flex-wrap:wrap;gap:var(--size-large);justify-content:flex-start;align-items:baseline}.callout-content{display:grid;padding:var(--size-medium) var(--size-xlarge);gap:var(--size-xsmall)}.callout-metric{display:flex;flex-direction:column;gap:var(--size-xsmall)}.callout-title{font-size:var(--type-3-font-size);line-height:var(--type-3-line-height-2)}.callout-label{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.callout-metric-value{font-size:var(--type-4-font-size);line-height:var(--type-4-line-height-1)}\n"] }]
270
+ args: [{ selector: 'riv-stacked-row', 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 draw\"\n [attr.viewBox]=\"draw.viewBox\"\n >\n <defs>\n <pattern\n id=\"stripes\"\n x=\"0\"\n y=\"0\"\n [attr.width]=\"draw.rowHeight\"\n [attr.height]=\"draw.rowHeight\"\n patternUnits=\"userSpaceOnUse\"\n >\n <line\n [attr.x1]=\"0\"\n [attr.y1]=\"0\"\n [attr.x2]=\"draw.rowHeight\"\n [attr.y2]=\"draw.rowHeight\"\n stroke=\"var(--white-100)\"\n stroke-width=\"1\"\n ></line>\n </pattern>\n </defs>\n <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n <text\n rivSVGTextTruncate\n [text]=\"rowNameToDraw.label\"\n [width]=\"SECTION_WIDTHS.ROW_NAME\"\n class=\"row-title\"\n [attr.x]=\"rowNameToDraw.x\"\n [attr.y]=\"rowNameToDraw.y\"\n ></text>\n </g>\n <g *ngFor=\"let tick of draw.xTicks\">\n <rect\n class=\"tick\"\n [attr.x]=\"draw.xScale(tick)\"\n y=\"0\"\n width=\"1\"\n height=\"100%\"\n ></rect>\n <text\n class=\"tick-label\"\n [attr.x]=\"draw.xScale(tick)\"\n [attr.y]=\"locationOfXAxisTick()\"\n dx=\"4\"\n dy=\"-4\"\n >\n {{ valueFormatter(tick) }}\n </text>\n </g>\n <g\n *ngFor=\"let rect of draw.rects\"\n (mouseenter)=\"hoveredBand$.next({ rowName: rect.rowName, event: $event })\"\n (mouseleave)=\"hoveredBand$.next(null)\"\n >\n <rect\n class=\"row\"\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)?.rowName !== rect.rowName\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 <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n <text\n class=\"row-total\"\n [attr.x]=\"rowTotalToDraw.x\"\n [attr.y]=\"rowTotalToDraw.y\"\n text-anchor=\"end\"\n >\n {{ rowTotalToDraw.label }}\n </text>\n </g>\n </svg>\n <legend *ngIf=\"drawData$ | async; let draw\">\n <riv-legend-item\n *ngFor=\"let item of draw.legendItems\"\n [label]=\"item.label\"\n [colorToken]=\"item.colorToken\"\n [style]=\"item.style ? item.style : 'solid'\"\n ></riv-legend-item>\n </legend>\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]=\"'bottom-center'\"\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 [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n", styles: [".tick{fill:var(--gray-20)}.tick-label{font-size:var(--type-0-font-size);line-height:var(--type-0-line-height-0);fill:var(--type-light-low-contrast)}.row-title{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2)}.row-total{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-2);fill:var(--type-light-low-contrast)}.row{transition:opacity var(--short-transition)}.row.blurred{opacity:.4}legend{display:flex;flex-wrap:wrap;gap:var(--size-large);justify-content:flex-start;align-items:baseline}.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"] }]
223
271
  }], propDecorators: { input: [{
224
272
  type: Input
225
273
  }], width: [{
@@ -231,4 +279,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
231
279
  }], zeroStateMessage: [{
232
280
  type: Input
233
281
  }] } });
234
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-row.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { InternSet } from 'd3-array';\nimport { ScaleLinear, scaleLinear } from 'd3-scale';\nimport {\n  BehaviorSubject,\n  Observable,\n  combineLatest,\n  map,\n  shareReplay,\n} from 'rxjs';\nimport {\n  RowDisplayInfo,\n  RowLabel,\n  RowRectangle,\n  StackedRowDrawData,\n  getKeyOrder,\n  getLegendItems,\n  getRowDisplayInfo,\n  standardizeRowStacks,\n} from './stacked-row.helpers';\n\ntype CalloutMetric = {\n  rowName: string;\n  elementLabel: string;\n  value: string;\n};\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-row',\n  templateUrl: './stacked-row.component.html',\n  styleUrls: ['./stacked-row.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedRowComponent {\n  // This is the list of rows we will display, in the order we will display them.\n  private readonly input$ = new BehaviorSubject<StackedRowComponent.RowStack[]>(\n    [],\n  );\n  @Input()\n  public set input(v: StackedRowComponent.RowStack[]) {\n    this.input$.next(v);\n  }\n  public get input(): StackedRowComponent.RowStack[] {\n    return this.input$.getValue();\n  }\n\n  // Width helps determine the size of the viewbox this graph is displaying in.\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  // Height helps determine the size of the viewbox this graph is displaying in.\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  // ValueFormatter determines how we style the values in a metric and the x-axis\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  private readonly zeroStateMessage$ = new BehaviorSubject<string>(\n    'No data to display.',\n  );\n  @Input()\n  public set zeroStateMessage(v: string) {\n    this.zeroStateMessage$.next(v);\n  }\n  public get zeroStateMessage(): string {\n    return this.zeroStateMessage$.getValue();\n  }\n\n  /*\n    The layout we are shooting for is:\n\n              |   X Axis Labels  |\n    Row Name  |   Rectangles     |   Totals\n  */\n  readonly SECTION_HEIGHTS = {\n    // Space reserved for the x-axis labels\n    X_AXIS_LABEL: 14,\n    // Height of each row\n    PER_ROW: 20,\n    // Heigh of each rectangle\n    PER_RECTANGLE: 8,\n  };\n\n  readonly SECTION_WIDTHS = {\n    // Space reserved for the name at the start of the row\n    ROW_NAME: 100,\n    // Space reserved for the totals at the end of the row\n    ROW_TOTAL_LABEL: 72,\n  };\n\n  readonly SECTION_PADDING = {\n    // Padding between each edge of the viewbox and the content\n    OUTER_TOP_BOTTOM: 4,\n    OUTER_LEFT_RIGHT: 4,\n    // Padding between the bottom of one row and the top of the next row\n    BETWEEN_ROW: 4,\n    // Space between the row and the top of a rectangle in that row\n    RECTANGLE_TO_ROW: 6,\n  };\n\n  // The number of X-axis segments to create, with labels at the start of each segment.\n  readonly X_TICK_COUNT = 8;\n\n  // Zero on the X axis is after the left padding and the row name.\n  locationOfXAxisZero() {\n    return this.SECTION_PADDING.OUTER_LEFT_RIGHT + this.SECTION_WIDTHS.ROW_NAME;\n  }\n  // The end of the X axis is right before the row total and the right padding.\n  locationOfXAxisEnd(width: number) {\n    const spaceOnRight: number =\n      this.SECTION_WIDTHS.ROW_TOTAL_LABEL +\n      this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n    return width - spaceOnRight;\n  }\n  // The y-index of the x-axis ticks\n  locationOfXAxisTick() {\n    return (\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL\n    );\n  }\n  // The y-index of the top of the current row\n  getTopOfRow(rowIndex: number) {\n    const spaceAboveRows: number =\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL;\n\n    const aboveRows: number = rowIndex * this.SECTION_HEIGHTS.PER_ROW;\n    const aboveRowPaddings: number =\n      rowIndex * this.SECTION_PADDING.BETWEEN_ROW;\n    const allAboveRows: number = aboveRows + aboveRowPaddings;\n\n    return spaceAboveRows + allAboveRows;\n  }\n\n  // Calculate all the values we need for displaying the graph on screen.\n  getDrawnData(\n    rowStacks: StackedRowComponent.RowStack[],\n    width: number,\n    height: number,\n  ) {\n    if (rowStacks.length == 0) {\n      return null;\n    }\n    const viewBox = `0 0 ${width} ${height}`;\n\n    const keysInOrder: InternSet<string> = getKeyOrder(rowStacks);\n    const standardizedRows: StackedRowComponent.RowStack[] =\n      standardizeRowStacks(rowStacks, keysInOrder);\n    const rowsForDisplay: RowDisplayInfo[] = standardizedRows.map(stackData => {\n      return getRowDisplayInfo(stackData);\n    });\n\n    // The maximum value of the domain should be the highest total value from the rows.\n    const domainMax = Math.max(0, ...rowsForDisplay.map(row => row.rowTotal));\n    const xScale: ScaleLinear<number, number, never> = scaleLinear()\n      .domain([0, domainMax])\n      .range([this.locationOfXAxisZero(), this.locationOfXAxisEnd(width)]);\n    const xTicks = xScale.ticks(this.X_TICK_COUNT);\n\n    const rects: RowRectangle[] = this.createRectanglesFromRows(\n      rowsForDisplay,\n      xScale,\n    );\n\n    const rowNamesToDraw: RowLabel[] = this.getRowNamesToDraw(rowsForDisplay);\n    const rowTotalsToDraw: RowLabel[] = this.getRowTotalsToDraw(\n      rowsForDisplay,\n      width,\n    );\n\n    const legendItems = getLegendItems(rowsForDisplay, keysInOrder);\n\n    return {\n      legendItems: legendItems,\n      rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE,\n      rects: rects,\n      rowNames: rowNamesToDraw,\n      rowTotals: rowTotalsToDraw,\n      viewBox: viewBox,\n      xScale: xScale,\n      xTicks: xTicks,\n    };\n  }\n\n  // Get data for all the rectangles for all the rows\n  createRectanglesFromRows(\n    displayRows: RowDisplayInfo[],\n    xScale: ScaleLinear<number, number, never>,\n  ): RowRectangle[] {\n    const rectangles: RowRectangle[] = [];\n    displayRows.forEach((currentRow, rowIndex) => {\n      currentRow.elementsButStacked.forEach((currentElement, _) => {\n        const topOfRow: number = this.getTopOfRow(rowIndex);\n\n        const rectangleWidth: number =\n          xScale(currentElement.topline) - xScale(currentElement.baseline);\n        const rectangleHeight: number = this.SECTION_HEIGHTS.PER_RECTANGLE;\n\n        const rectangleX: number = xScale(currentElement.baseline);\n        const rectangleY: number =\n          topOfRow + this.SECTION_PADDING.RECTANGLE_TO_ROW;\n\n        const fill = `var(${currentElement.colorToken})`;\n\n        rectangles.push({\n          x: rectangleX,\n          y: rectangleY,\n          width: rectangleWidth,\n          height: rectangleHeight,\n          fill: fill,\n          striped: currentElement.style === 'striped',\n          rowName: currentRow.rowName,\n          stackedRowElement: currentElement,\n        });\n      });\n    });\n    return rectangles;\n  }\n\n  // Get data on the row names we are drawing\n  getRowNamesToDraw(displayRows: RowDisplayInfo[]): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row name looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE;\n\n      return {\n        label: row.rowName,\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  // Get data on the row totals we are drawing\n  getRowTotalsToDraw(displayRows: RowDisplayInfo[], width: number): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = width - this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row totals looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE;\n\n      return {\n        label: this.valueFormatter(row.rowTotal),\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  readonly drawData$: Observable<StackedRowDrawData | null> = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      return this.getDrawnData(input, width, height);\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // Lets us know whether to display the zero data state\n  readonly empty$ = this.input$.pipe(\n    map(input => input.every(stack => stack.data.length === 0)),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // What rectangle we are currently hovering over\n  readonly hoveredRectangle$ = new BehaviorSubject<{\n    rectangle: RowRectangle;\n    event: MouseEvent;\n  } | null>(null);\n\n  // Create the metric to display based on the currently hovered rectangle\n  readonly callout$ = this.hoveredRectangle$.pipe(\n    map(hoveredRectangle => {\n      if (!hoveredRectangle) return null;\n\n      const { target } = hoveredRectangle.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      const { rowName, stackedRowElement } = hoveredRectangle.rectangle;\n\n      const metric: CalloutMetric = {\n        rowName: rowName,\n        elementLabel: stackedRowElement.label,\n        value: this.valueFormatter(stackedRowElement.value),\n      };\n\n      return { anchor, metric };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n\nexport namespace StackedRowComponent {\n  // A specifically colored section of one row.\n  export type RowElement = {\n    label: string; // Legend element name\n    value: number;\n    colorToken: string;\n    style?: 'solid' | 'striped';\n  };\n  // The input that determines one row.\n  export type RowStack = {\n    rowName: string;\n    data: RowElement[];\n  };\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 draw\"\n    [attr.viewBox]=\"draw.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"draw.rowHeight\"\n        [attr.height]=\"draw.rowHeight\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          [attr.x1]=\"0\"\n          [attr.y1]=\"0\"\n          [attr.x2]=\"draw.rowHeight\"\n          [attr.y2]=\"draw.rowHeight\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n      <text\n        rivSVGTextTruncate\n        [text]=\"rowNameToDraw.label\"\n        [width]=\"SECTION_WIDTHS.ROW_NAME\"\n        class=\"row-title\"\n        [attr.x]=\"rowNameToDraw.x\"\n        [attr.y]=\"rowNameToDraw.y\"\n      ></text>\n    </g>\n    <g *ngFor=\"let tick of draw.xTicks\">\n      <rect\n        class=\"tick\"\n        [attr.x]=\"draw.xScale(tick)\"\n        y=\"0\"\n        width=\"1\"\n        height=\"100%\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"draw.xScale(tick)\"\n        [attr.y]=\"locationOfXAxisTick()\"\n        dx=\"4\"\n        dy=\"-4\"\n      >\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g *ngFor=\"let rect of draw.rects\">\n      <rect\n        class=\"row\"\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          (hoveredRectangle$ | async) !== null &&\n          (hoveredRectangle$ | async)?.rectangle?.rowName !== rect.rowName\n        \"\n        (mouseenter)=\"\n          hoveredRectangle$.next({ rectangle: rect, event: $event })\n        \"\n        (mouseleave)=\"hoveredRectangle$.next(null)\"\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        (mouseenter)=\"\n          hoveredRectangle$.next({ rectangle: rect, event: $event })\n        \"\n        (mouseleave)=\"hoveredRectangle$.next(null)\"\n      ></rect>\n    </g>\n    <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n      <text\n        class=\"row-total\"\n        [attr.x]=\"rowTotalToDraw.x\"\n        [attr.y]=\"rowTotalToDraw.y\"\n        text-anchor=\"end\"\n      >\n        {{ rowTotalToDraw.label }}\n      </text>\n    </g>\n  </svg>\n  <legend *ngIf=\"drawData$ | async; let draw\">\n    <riv-legend-item\n      *ngFor=\"let item of draw.legendItems\"\n      [label]=\"item.label\"\n      [colorToken]=\"item.colorToken\"\n      [style]=\"item.style ? item.style : 'solid'\"\n    ></riv-legend-item>\n  </legend>\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-title\">{{ callout.metric.rowName }}</div>\n      <div class=\"callout-metric\">\n        <div class=\"callout-label\">{{ callout.metric.elementLabel }}</div>\n        <div class=\"callout-metric-value\">{{ callout.metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n"]}
282
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-row.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { InternSet } from 'd3-array';\nimport { ScaleLinear, scaleLinear } from 'd3-scale';\nimport {\n  BehaviorSubject,\n  Observable,\n  combineLatest,\n  map,\n  shareReplay,\n} from 'rxjs';\nimport {\n  RowDisplayInfo,\n  RowLabel,\n  RowRectangle,\n  StackedRowDrawData,\n  getKeyOrder,\n  getLegendItems,\n  getRowDisplayInfo,\n  standardizeRowStacks,\n} from './stacked-row.helpers';\n\ntype CalloutMetric = {\n  rowName: string;\n  elementLabel: string;\n  value: string;\n};\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-row',\n  templateUrl: './stacked-row.component.html',\n  styleUrls: ['./stacked-row.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedRowComponent {\n  // This is the list of rows we will display, in the order we will display them.\n  private readonly input$ = new BehaviorSubject<StackedRowComponent.RowStack[]>(\n    [],\n  );\n  @Input()\n  public set input(v: StackedRowComponent.RowStack[]) {\n    this.input$.next(v);\n  }\n  public get input(): StackedRowComponent.RowStack[] {\n    return this.input$.getValue();\n  }\n\n  // Width helps determine the size of the viewbox this graph is displaying in.\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  // Height helps determine the size of the viewbox this graph is displaying in.\n  private readonly height$ = new BehaviorSubject<number | null>(null);\n  @Input()\n  public set height(v: number | null) {\n    this.height$.next(v);\n  }\n  public get height(): number | null {\n    return this.height$.getValue();\n  }\n\n  // ValueFormatter determines how we style the values in a metric and the x-axis\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  private readonly zeroStateMessage$ = new BehaviorSubject<string>(\n    'No data to display.',\n  );\n  @Input()\n  public set zeroStateMessage(v: string) {\n    this.zeroStateMessage$.next(v);\n  }\n  public get zeroStateMessage(): string {\n    return this.zeroStateMessage$.getValue();\n  }\n\n  /*\n    The layout we are shooting for is:\n\n              |   X Axis Labels  |\n    Row Name  |   Rectangles     |   Totals\n  */\n  readonly RECTANGLE_HEIGHTS = {\n    MIN_HEIGHT: 12, // Sets the smallest size a row can shrink to\n    MAX_HEIGHT: 42, // Sets the largest size a row can expand to\n  };\n\n  readonly SECTION_HEIGHTS = {\n    // Space reserved for the x-axis labels\n    X_AXIS_LABEL: 14,\n    // Height of each row\n    PER_ROW: (): number => {\n      //Only use height if specified\n      if (this.input.length && this.height) {\n        let rowHeight =\n          (this.height -\n            this.SECTION_HEIGHTS.X_AXIS_LABEL -\n            this.SECTION_PADDING.OUTER_TOP_BOTTOM -\n            this.SECTION_PADDING.RECTANGLE_TO_ROW) /\n            this.input.length -\n          this.SECTION_PADDING.BETWEEN_ROW; // calculate base row height\n        // Clamping the row height between the min and max heights\n        return Math.min(\n          Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT),\n          this.RECTANGLE_HEIGHTS.MAX_HEIGHT,\n        );\n      }\n      return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input\n    },\n    // Max Height of each rectangle\n    PER_RECTANGLE: (): number => {\n      //Only use height if specified\n      if (this.input.length && this.height) {\n        let rectangleHeight =\n          (this.height -\n            this.SECTION_HEIGHTS.X_AXIS_LABEL -\n            this.SECTION_PADDING.OUTER_TOP_BOTTOM -\n            this.SECTION_PADDING.RECTANGLE_TO_ROW) /\n            this.input.length -\n          this.SECTION_PADDING.BETWEEN_ROW; // calculate base rectangle height\n        // Clamping the rectangle height between the min and max heights\n        return Math.min(\n          Math.max(rectangleHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT),\n          this.RECTANGLE_HEIGHTS.MAX_HEIGHT,\n        );\n      }\n      return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input\n    },\n  };\n\n  readonly SECTION_WIDTHS = {\n    // Space reserved for the name at the start of the row\n    ROW_NAME: 100,\n    // Space reserved for the totals at the end of the row\n    ROW_TOTAL_LABEL: 72,\n  };\n\n  readonly SECTION_PADDING = {\n    // Padding between each edge of the viewbox and the content\n    OUTER_TOP_BOTTOM: 4,\n    OUTER_LEFT_RIGHT: 4,\n    // Padding between the bottom of one row and the top of the next row\n    BETWEEN_ROW: 4,\n    // Space between the row and the top of a rectangle in that row\n    RECTANGLE_TO_ROW: 6,\n  };\n\n  // The number of X-axis segments to create, with labels at the start of each segment.\n  readonly X_TICK_COUNT_MAX = 8;\n  readonly X_TICK_COUNT_MIN = 3;\n  readonly X_TICK_MAX_WIDTH = 900;\n  readonly X_TICK_MIN_WIDTH = 400;\n\n  // Zero on the X axis is after the left padding and the row name.\n  locationOfXAxisZero() {\n    return this.SECTION_PADDING.OUTER_LEFT_RIGHT + this.SECTION_WIDTHS.ROW_NAME;\n  }\n  // The end of the X axis is right before the row total and the right padding.\n  locationOfXAxisEnd(width: number) {\n    const spaceOnRight: number =\n      this.SECTION_WIDTHS.ROW_TOTAL_LABEL +\n      this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n    return width - spaceOnRight;\n  }\n  // The y-index of the x-axis ticks\n  locationOfXAxisTick() {\n    return (\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL\n    );\n  }\n  // The y-index of the top of the current row\n  getTopOfRow(rowIndex: number) {\n    const spaceAboveRows: number =\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL;\n\n    const aboveRows: number = rowIndex * this.SECTION_HEIGHTS.PER_ROW();\n    const aboveRowPaddings: number =\n      rowIndex * this.SECTION_PADDING.BETWEEN_ROW;\n    const allAboveRows: number = aboveRows + aboveRowPaddings;\n\n    return spaceAboveRows + allAboveRows;\n  }\n\n  // Calculate all the values we need for displaying the graph on screen.\n  getDrawnData(\n    rowStacks: StackedRowComponent.RowStack[],\n    width: number,\n    height: number,\n  ) {\n    if (rowStacks.length == 0) {\n      return null;\n    }\n    const viewBox = `0 0 ${width} ${height}`;\n\n    const keysInOrder: InternSet<string> = getKeyOrder(rowStacks);\n    const standardizedRows: StackedRowComponent.RowStack[] =\n      standardizeRowStacks(rowStacks, keysInOrder);\n    const rowsForDisplay: RowDisplayInfo[] = standardizedRows.map(stackData => {\n      return getRowDisplayInfo(stackData);\n    });\n\n    // The maximum value of the domain should be the highest total value from the rows.\n    const domainMax = Math.max(0, ...rowsForDisplay.map(row => row.rowTotal));\n    const xScale: ScaleLinear<number, number, never> = scaleLinear()\n      .domain([0, domainMax])\n      .range([this.locationOfXAxisZero(), this.locationOfXAxisEnd(width)]);\n\n    // Setup a linear scale for tick count based on width\n    const tickScale = scaleLinear()\n      .domain([this.X_TICK_MIN_WIDTH, this.X_TICK_MAX_WIDTH])\n      .range([this.X_TICK_COUNT_MIN, this.X_TICK_COUNT_MAX])\n      .clamp(true); // Ensures the scale does not go beyond the defined range\n\n    const xtickCountAdjusted = Math.round(tickScale(width));\n    const xTicks = xScale.ticks(xtickCountAdjusted);\n\n    const rects: RowRectangle[] = this.createRectanglesFromRows(\n      rowsForDisplay,\n      xScale,\n    );\n\n    const rowNamesToDraw: RowLabel[] = this.getRowNamesToDraw(rowsForDisplay);\n    const rowTotalsToDraw: RowLabel[] = this.getRowTotalsToDraw(\n      rowsForDisplay,\n      width,\n    );\n\n    const legendItems = getLegendItems(rowsForDisplay, keysInOrder);\n\n    return {\n      legendItems: legendItems,\n      rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE(),\n      rects: rects,\n      rowNames: rowNamesToDraw,\n      rowTotals: rowTotalsToDraw,\n      viewBox: viewBox,\n      xScale: xScale,\n      xTicks: xTicks,\n    };\n  }\n\n  // Get data for all the rectangles for all the rows\n  createRectanglesFromRows(\n    displayRows: RowDisplayInfo[],\n    xScale: ScaleLinear<number, number, never>,\n  ): RowRectangle[] {\n    const rectangles: RowRectangle[] = [];\n    displayRows.forEach((currentRow, rowIndex) => {\n      currentRow.elementsButStacked.forEach((currentElement, _) => {\n        const topOfRow: number = this.getTopOfRow(rowIndex);\n\n        const rectangleWidth: number =\n          xScale(currentElement.topline) - xScale(currentElement.baseline);\n        const rectangleHeight: number = this.SECTION_HEIGHTS.PER_RECTANGLE();\n\n        const rectangleX: number = xScale(currentElement.baseline);\n        const rectangleY: number =\n          topOfRow + this.SECTION_PADDING.RECTANGLE_TO_ROW;\n\n        const fill = `var(${currentElement.colorToken})`;\n\n        rectangles.push({\n          x: rectangleX,\n          y: rectangleY,\n          width: rectangleWidth,\n          height: rectangleHeight,\n          fill: fill,\n          striped: currentElement.style === 'striped',\n          rowName: currentRow.rowName,\n          stackedRowElement: currentElement,\n        });\n      });\n    });\n    return rectangles;\n  }\n\n  // Get data on the row names we are drawing\n  getRowNamesToDraw(displayRows: RowDisplayInfo[]): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row name looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;\n\n      return {\n        label: row.rowName,\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  // Get data on the row totals we are drawing\n  getRowTotalsToDraw(displayRows: RowDisplayInfo[], width: number): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = width - this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row totals looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;\n\n      return {\n        label: this.valueFormatter(row.rowTotal),\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  readonly drawData$: Observable<StackedRowDrawData | null> = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      return this.getDrawnData(\n        input,\n        width,\n        height\n          ? height\n          : this.SECTION_HEIGHTS.X_AXIS_LABEL +\n              this.SECTION_PADDING.OUTER_TOP_BOTTOM +\n              this.SECTION_PADDING.RECTANGLE_TO_ROW +\n              (this.SECTION_HEIGHTS.PER_ROW() +\n                this.SECTION_PADDING.BETWEEN_ROW) *\n                input.length,\n      );\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // Lets us know whether to display the zero data state\n  readonly empty$ = this.input$.pipe(\n    map(input => input.every(stack => stack.data.length === 0)),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // What rectangle we are currently hovering over\n  readonly hoveredRectangle$ = new BehaviorSubject<{\n    rectangle: RowRectangle;\n    event: MouseEvent;\n  } | null>(null);\n\n  // What band we are currently hovering over\n  readonly hoveredBand$ = new BehaviorSubject<{\n    rowName: string;\n    event: MouseEvent;\n  } | null>(null);\n\n  // Create the metric to display based on the currently hovered rectangle\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand || !drawData) 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.rowName === hoveredBand.rowName)\n        .map(rect => ({\n          label: rect.stackedRowElement.label,\n          value: this.valueFormatter(rect.stackedRowElement.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n\nexport namespace StackedRowComponent {\n  // A specifically colored section of one row.\n  export type RowElement = {\n    label: string; // Legend element name\n    value: number;\n    colorToken: string;\n    style?: 'solid' | 'striped';\n  };\n  // The input that determines one row.\n  export type RowStack = {\n    rowName: string;\n    data: RowElement[];\n  };\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 draw\"\n    [attr.viewBox]=\"draw.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"draw.rowHeight\"\n        [attr.height]=\"draw.rowHeight\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          [attr.x1]=\"0\"\n          [attr.y1]=\"0\"\n          [attr.x2]=\"draw.rowHeight\"\n          [attr.y2]=\"draw.rowHeight\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n      <text\n        rivSVGTextTruncate\n        [text]=\"rowNameToDraw.label\"\n        [width]=\"SECTION_WIDTHS.ROW_NAME\"\n        class=\"row-title\"\n        [attr.x]=\"rowNameToDraw.x\"\n        [attr.y]=\"rowNameToDraw.y\"\n      ></text>\n    </g>\n    <g *ngFor=\"let tick of draw.xTicks\">\n      <rect\n        class=\"tick\"\n        [attr.x]=\"draw.xScale(tick)\"\n        y=\"0\"\n        width=\"1\"\n        height=\"100%\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"draw.xScale(tick)\"\n        [attr.y]=\"locationOfXAxisTick()\"\n        dx=\"4\"\n        dy=\"-4\"\n      >\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g\n      *ngFor=\"let rect of draw.rects\"\n      (mouseenter)=\"hoveredBand$.next({ rowName: rect.rowName, event: $event })\"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    >\n      <rect\n        class=\"row\"\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)?.rowName !== rect.rowName\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    <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n      <text\n        class=\"row-total\"\n        [attr.x]=\"rowTotalToDraw.x\"\n        [attr.y]=\"rowTotalToDraw.y\"\n        text-anchor=\"end\"\n      >\n        {{ rowTotalToDraw.label }}\n      </text>\n    </g>\n  </svg>\n  <legend *ngIf=\"drawData$ | async; let draw\">\n    <riv-legend-item\n      *ngFor=\"let item of draw.legendItems\"\n      [label]=\"item.label\"\n      [colorToken]=\"item.colorToken\"\n      [style]=\"item.style ? item.style : 'solid'\"\n    ></riv-legend-item>\n  </legend>\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]=\"'bottom-center'\"\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 [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n"]}