@rivet-health/design-system 2.14.0 → 2.15.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.
- package/esm2020/lib/input/single-select/single-select.component.mjs +2 -2
- package/esm2020/lib/visualization/intervals.mjs +63 -0
- package/esm2020/lib/visualization/stacked-column/stacked-column.component.mjs +129 -25
- package/esm2020/lib/visualization/time-series/time-series.component.mjs +66 -79
- package/fesm2015/rivet-health-design-system.mjs +251 -99
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +249 -99
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/lib/visualization/intervals.d.ts +8 -0
- package/lib/visualization/stacked-column/stacked-column.component.d.ts +24 -6
- package/lib/visualization/time-series/time-series.component.d.ts +13 -8
- package/package.json +1 -1
|
@@ -37,10 +37,10 @@ export class SingleSelectComponent {
|
|
|
37
37
|
}
|
|
38
38
|
}
|
|
39
39
|
SingleSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SingleSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
40
|
-
SingleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SingleSelectComponent, selector: "riv-single-select", inputs: { options: "options", selectedOption: "selectedOption", filterabilityOptions: "filterabilityOptions", loading: "loading", noOptionsMessage: "noOptionsMessage", nodeTemplate: "nodeTemplate", placeholder: "placeholder" }, outputs: { filterQueryChange: "filterQueryChange", selectedOptionChange: "selectedOptionChange" }, ngImport: i0, template: "<button #trigger class=\"trigger\" (click)=\"open = true\">\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n </span>\n</button>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"trigger\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n />\n <div class=\"options\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngIf=\"getNodes(); let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node: node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.selected\"\n [class.disabled]=\"node?.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.selected\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </riv-loading-cover>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-height:calc(var(--base-grid-size) * 200);max-width:calc(var(--base-grid-size) * 150);overflow-y:
|
|
40
|
+
SingleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SingleSelectComponent, selector: "riv-single-select", inputs: { options: "options", selectedOption: "selectedOption", filterabilityOptions: "filterabilityOptions", loading: "loading", noOptionsMessage: "noOptionsMessage", nodeTemplate: "nodeTemplate", placeholder: "placeholder" }, outputs: { filterQueryChange: "filterQueryChange", selectedOptionChange: "selectedOptionChange" }, ngImport: i0, template: "<button #trigger class=\"trigger\" (click)=\"open = true\">\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n </span>\n</button>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"trigger\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n />\n <div class=\"options\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngIf=\"getNodes(); let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node: node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.selected\"\n [class.disabled]=\"node?.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.selected\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </riv-loading-cover>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-height:calc(var(--base-grid-size) * 200);max-width:calc(var(--base-grid-size) * 150);overflow-y:auto}.single-select-node{display:flex;align-items:center;overflow:hidden;flex-grow:1;text-align:left;padding:var(--size-xsmall) var(--size-xsmall) var(--size-xsmall) calc(var(--base-grid-size) * 6);width:100%}.single-select-node.selected{padding-left:var(--size-small)}.single-select-node:hover{background-color:var(--surface-light-2)}.single-select-node.disabled{cursor:default;background-color:var(--surface-light-0)}.single-select-node.disabled .label,.single-select-node.disabled .label-subtitle{color:var(--type-light-disabled)}.single-select-node riv-icon{flex-shrink:0}.single-select-node .label{font-size:calc(var(--base-grid-size) * 4);line-height:calc(var(--base-grid-size) * 6);overflow:hidden;text-overflow:ellipsis;white-space:pre;flex-grow:1;padding:0 calc(var(--base-grid-size) * 2)}.single-select-node .label-subtitle{padding-left:var(--base-grid-size);color:var(--type-light-low-contrast)}.custom-single-select-node{display:block;width:100%;text-align:left}.empty{padding:var(--size-medium);text-align:center;color:var(--type-light-disabled)}\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: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "component", type: i3.HighlightComponent, selector: "riv-highlight", inputs: ["text", "indices"] }, { kind: "component", type: i4.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "component", type: i5.LoadingCoverComponent, selector: "riv-loading-cover", inputs: ["loading", "loadingSize", "errorMessage"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
41
41
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SingleSelectComponent, decorators: [{
|
|
42
42
|
type: Component,
|
|
43
|
-
args: [{ selector: 'riv-single-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<button #trigger class=\"trigger\" (click)=\"open = true\">\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n </span>\n</button>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"trigger\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n />\n <div class=\"options\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngIf=\"getNodes(); let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node: node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.selected\"\n [class.disabled]=\"node?.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.selected\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </riv-loading-cover>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-height:calc(var(--base-grid-size) * 200);max-width:calc(var(--base-grid-size) * 150);overflow-y:
|
|
43
|
+
args: [{ selector: 'riv-single-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<button #trigger class=\"trigger\" (click)=\"open = true\">\n <ng-container *ngIf=\"selectedOption; else placeholderValue\">\n <span class=\"value\">{{ selectedOption.title }}</span>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n </span>\n</button>\n\n<riv-callout\n *ngIf=\"open\"\n [anchor]=\"trigger\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"open = false\"\n>\n <input\n *ngIf=\"filterabilityOptions.enabled\"\n #filter\n class=\"filter\"\n [placeholder]=\"getFilterPlaceholder()\"\n [value]=\"filterQuery\"\n (input)=\"filterQuery = filter.value; filterQueryChange.emit(filterQuery)\"\n />\n <div class=\"options\">\n <riv-loading-cover [loading]=\"loading\">\n <ng-container *ngIf=\"getNodes(); let nodes\">\n <ng-container *ngIf=\"nodes.length > 0; else empty\">\n <ng-container *ngFor=\"let node of nodes\">\n <ng-container *ngIf=\"nodeTemplate; else standardTemplate\">\n <button\n class=\"custom-single-select-node\"\n [disabled]=\"node.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate\"\n [ngTemplateOutletContext]=\"{ node: node }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTemplate>\n <button\n class=\"single-select-node\"\n [class.selected]=\"node?.selected\"\n [class.disabled]=\"node?.disabled\"\n (click)=\"selectedOptionChange.emit(node); open = false\"\n >\n <riv-icon\n [name]=\"'Check'\"\n *ngIf=\"node?.selected\"\n [size]=\"16\"\n ></riv-icon>\n <span class=\"label\">\n <span class=\"label-title\">\n <riv-highlight\n [text]=\"node?.title || ''\"\n [indices]=\"node?.titleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n <span *ngIf=\"node?.subtitle\" class=\"label-subtitle\">\n <riv-highlight\n [text]=\"node?.subtitle || ''\"\n [indices]=\"node?.subtitleHighlightIndices || []\"\n ></riv-highlight>\n </span>\n </span>\n </button>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #empty>\n <div class=\"empty\">\n {{ noOptionsMessage }}\n </div>\n </ng-template>\n </ng-container>\n </riv-loading-cover>\n </div>\n</riv-callout>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1)}.value{font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small);flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-small);background-color:var(--surface-light-2);border-left:var(--border-width) solid var(--border-light)}.filter{width:100%;outline:none;border:none;border-bottom:var(--border-width) solid var(--border-light);font-size:var(--type-2-font-size);line-height:var(--type-2-line-height-0);color:var(--type-light-high-contrast);padding:var(--size-small)}.filter::placeholder{color:var(--type-light-disabled)}.options{max-height:calc(var(--base-grid-size) * 200);max-width:calc(var(--base-grid-size) * 150);overflow-y:auto}.single-select-node{display:flex;align-items:center;overflow:hidden;flex-grow:1;text-align:left;padding:var(--size-xsmall) var(--size-xsmall) var(--size-xsmall) calc(var(--base-grid-size) * 6);width:100%}.single-select-node.selected{padding-left:var(--size-small)}.single-select-node:hover{background-color:var(--surface-light-2)}.single-select-node.disabled{cursor:default;background-color:var(--surface-light-0)}.single-select-node.disabled .label,.single-select-node.disabled .label-subtitle{color:var(--type-light-disabled)}.single-select-node riv-icon{flex-shrink:0}.single-select-node .label{font-size:calc(var(--base-grid-size) * 4);line-height:calc(var(--base-grid-size) * 6);overflow:hidden;text-overflow:ellipsis;white-space:pre;flex-grow:1;padding:0 calc(var(--base-grid-size) * 2)}.single-select-node .label-subtitle{padding-left:var(--base-grid-size);color:var(--type-light-low-contrast)}.custom-single-select-node{display:block;width:100%;text-align:left}.empty{padding:var(--size-medium);text-align:center;color:var(--type-light-disabled)}\n"] }]
|
|
44
44
|
}], propDecorators: { options: [{
|
|
45
45
|
type: Input
|
|
46
46
|
}], selectedOption: [{
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import { timeDay, timeMonth, timeWeek, timeYear } from 'd3-time';
|
|
2
|
+
export const intervals = ['year', 'quarter', 'month', 'week', 'day'];
|
|
3
|
+
export function getTimeInterval(interval) {
|
|
4
|
+
switch (interval) {
|
|
5
|
+
case 'day':
|
|
6
|
+
return timeDay;
|
|
7
|
+
case 'week':
|
|
8
|
+
return timeWeek;
|
|
9
|
+
case 'month':
|
|
10
|
+
return timeMonth;
|
|
11
|
+
case 'quarter':
|
|
12
|
+
return timeMonth.every(3);
|
|
13
|
+
case 'year':
|
|
14
|
+
return timeYear;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
export function getMinorIntervalFormat(interval) {
|
|
18
|
+
switch (interval) {
|
|
19
|
+
case 'day':
|
|
20
|
+
case 'week':
|
|
21
|
+
return '%-d';
|
|
22
|
+
case 'month':
|
|
23
|
+
return '%b';
|
|
24
|
+
case 'quarter':
|
|
25
|
+
return 'Q%q';
|
|
26
|
+
case 'year':
|
|
27
|
+
return '%Y';
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
export function getMajorInterval(interval) {
|
|
31
|
+
switch (interval) {
|
|
32
|
+
case 'day':
|
|
33
|
+
case 'week':
|
|
34
|
+
return 'month';
|
|
35
|
+
case 'month':
|
|
36
|
+
case 'quarter':
|
|
37
|
+
case 'year':
|
|
38
|
+
return 'year';
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
export function getMajorIntervalFormat(interval, hasMultipleYears) {
|
|
42
|
+
switch (interval) {
|
|
43
|
+
case 'year':
|
|
44
|
+
return hasMultipleYears ? '%Y' : '';
|
|
45
|
+
case 'month':
|
|
46
|
+
return hasMultipleYears ? '%b %Y' : '%b';
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
export function getIntervalTitle(interval) {
|
|
50
|
+
switch (interval) {
|
|
51
|
+
case 'year':
|
|
52
|
+
return 'Years';
|
|
53
|
+
case 'quarter':
|
|
54
|
+
return 'Quarters';
|
|
55
|
+
case 'month':
|
|
56
|
+
return 'Months';
|
|
57
|
+
case 'week':
|
|
58
|
+
return 'Weeks';
|
|
59
|
+
case 'day':
|
|
60
|
+
return 'Days';
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJ2YWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvdmlzdWFsaXphdGlvbi9pbnRlcnZhbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFnQixPQUFPLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBRSxRQUFRLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFFL0UsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFHLENBQUMsTUFBTSxFQUFFLFNBQVMsRUFBRSxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBVSxDQUFDO0FBRzlFLE1BQU0sVUFBVSxlQUFlLENBQUMsUUFBa0I7SUFDaEQsUUFBUSxRQUFRLEVBQUU7UUFDaEIsS0FBSyxLQUFLO1lBQ1IsT0FBTyxPQUFPLENBQUM7UUFDakIsS0FBSyxNQUFNO1lBQ1QsT0FBTyxRQUFRLENBQUM7UUFDbEIsS0FBSyxPQUFPO1lBQ1YsT0FBTyxTQUFTLENBQUM7UUFDbkIsS0FBSyxTQUFTO1lBQ1osT0FBTyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBRSxDQUFDO1FBQzdCLEtBQUssTUFBTTtZQUNULE9BQU8sUUFBUSxDQUFDO0tBQ25CO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxzQkFBc0IsQ0FBQyxRQUFrQjtJQUN2RCxRQUFRLFFBQVEsRUFBRTtRQUNoQixLQUFLLEtBQUssQ0FBQztRQUNYLEtBQUssTUFBTTtZQUNULE9BQU8sS0FBSyxDQUFDO1FBQ2YsS0FBSyxPQUFPO1lBQ1YsT0FBTyxJQUFJLENBQUM7UUFDZCxLQUFLLFNBQVM7WUFDWixPQUFPLEtBQUssQ0FBQztRQUNmLEtBQUssTUFBTTtZQUNULE9BQU8sSUFBSSxDQUFDO0tBQ2Y7QUFDSCxDQUFDO0FBRUQsTUFBTSxVQUFVLGdCQUFnQixDQUFDLFFBQWtCO0lBQ2pELFFBQVEsUUFBUSxFQUFFO1FBQ2hCLEtBQUssS0FBSyxDQUFDO1FBQ1gsS0FBSyxNQUFNO1lBQ1QsT0FBTyxPQUFPLENBQUM7UUFDakIsS0FBSyxPQUFPLENBQUM7UUFDYixLQUFLLFNBQVMsQ0FBQztRQUNmLEtBQUssTUFBTTtZQUNULE9BQU8sTUFBTSxDQUFDO0tBQ2pCO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxzQkFBc0IsQ0FDcEMsUUFBNkMsRUFDN0MsZ0JBQXlCO0lBRXpCLFFBQVEsUUFBUSxFQUFFO1FBQ2hCLEtBQUssTUFBTTtZQUNULE9BQU8sZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO1FBQ3RDLEtBQUssT0FBTztZQUNWLE9BQU8sZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0tBQzVDO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxnQkFBZ0IsQ0FBQyxRQUFrQjtJQUNqRCxRQUFRLFFBQVEsRUFBRTtRQUNoQixLQUFLLE1BQU07WUFDVCxPQUFPLE9BQU8sQ0FBQztRQUNqQixLQUFLLFNBQVM7WUFDWixPQUFPLFVBQVUsQ0FBQztRQUNwQixLQUFLLE9BQU87WUFDVixPQUFPLFFBQVEsQ0FBQztRQUNsQixLQUFLLE1BQU07WUFDVCxPQUFPLE9BQU8sQ0FBQztRQUNqQixLQUFLLEtBQUs7WUFDUixPQUFPLE1BQU0sQ0FBQztLQUNqQjtBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUaW1lSW50ZXJ2YWwsIHRpbWVEYXksIHRpbWVNb250aCwgdGltZVdlZWssIHRpbWVZZWFyIH0gZnJvbSAnZDMtdGltZSc7XG5cbmV4cG9ydCBjb25zdCBpbnRlcnZhbHMgPSBbJ3llYXInLCAncXVhcnRlcicsICdtb250aCcsICd3ZWVrJywgJ2RheSddIGFzIGNvbnN0O1xuZXhwb3J0IHR5cGUgSW50ZXJ2YWwgPSAodHlwZW9mIGludGVydmFscylbbnVtYmVyXTtcblxuZXhwb3J0IGZ1bmN0aW9uIGdldFRpbWVJbnRlcnZhbChpbnRlcnZhbDogSW50ZXJ2YWwpOiBUaW1lSW50ZXJ2YWwge1xuICBzd2l0Y2ggKGludGVydmFsKSB7XG4gICAgY2FzZSAnZGF5JzpcbiAgICAgIHJldHVybiB0aW1lRGF5O1xuICAgIGNhc2UgJ3dlZWsnOlxuICAgICAgcmV0dXJuIHRpbWVXZWVrO1xuICAgIGNhc2UgJ21vbnRoJzpcbiAgICAgIHJldHVybiB0aW1lTW9udGg7XG4gICAgY2FzZSAncXVhcnRlcic6XG4gICAgICByZXR1cm4gdGltZU1vbnRoLmV2ZXJ5KDMpITtcbiAgICBjYXNlICd5ZWFyJzpcbiAgICAgIHJldHVybiB0aW1lWWVhcjtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0TWlub3JJbnRlcnZhbEZvcm1hdChpbnRlcnZhbDogSW50ZXJ2YWwpIHtcbiAgc3dpdGNoIChpbnRlcnZhbCkge1xuICAgIGNhc2UgJ2RheSc6XG4gICAgY2FzZSAnd2Vlayc6XG4gICAgICByZXR1cm4gJyUtZCc7XG4gICAgY2FzZSAnbW9udGgnOlxuICAgICAgcmV0dXJuICclYic7XG4gICAgY2FzZSAncXVhcnRlcic6XG4gICAgICByZXR1cm4gJ1ElcSc7XG4gICAgY2FzZSAneWVhcic6XG4gICAgICByZXR1cm4gJyVZJztcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0TWFqb3JJbnRlcnZhbChpbnRlcnZhbDogSW50ZXJ2YWwpIHtcbiAgc3dpdGNoIChpbnRlcnZhbCkge1xuICAgIGNhc2UgJ2RheSc6XG4gICAgY2FzZSAnd2Vlayc6XG4gICAgICByZXR1cm4gJ21vbnRoJztcbiAgICBjYXNlICdtb250aCc6XG4gICAgY2FzZSAncXVhcnRlcic6XG4gICAgY2FzZSAneWVhcic6XG4gICAgICByZXR1cm4gJ3llYXInO1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRNYWpvckludGVydmFsRm9ybWF0KFxuICBpbnRlcnZhbDogUmV0dXJuVHlwZTx0eXBlb2YgZ2V0TWFqb3JJbnRlcnZhbD4sXG4gIGhhc011bHRpcGxlWWVhcnM6IGJvb2xlYW4sXG4pIHtcbiAgc3dpdGNoIChpbnRlcnZhbCkge1xuICAgIGNhc2UgJ3llYXInOlxuICAgICAgcmV0dXJuIGhhc011bHRpcGxlWWVhcnMgPyAnJVknIDogJyc7XG4gICAgY2FzZSAnbW9udGgnOlxuICAgICAgcmV0dXJuIGhhc011bHRpcGxlWWVhcnMgPyAnJWIgJVknIDogJyViJztcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0SW50ZXJ2YWxUaXRsZShpbnRlcnZhbDogSW50ZXJ2YWwpIHtcbiAgc3dpdGNoIChpbnRlcnZhbCkge1xuICAgIGNhc2UgJ3llYXInOlxuICAgICAgcmV0dXJuICdZZWFycyc7XG4gICAgY2FzZSAncXVhcnRlcic6XG4gICAgICByZXR1cm4gJ1F1YXJ0ZXJzJztcbiAgICBjYXNlICdtb250aCc6XG4gICAgICByZXR1cm4gJ01vbnRocyc7XG4gICAgY2FzZSAnd2Vlayc6XG4gICAgICByZXR1cm4gJ1dlZWtzJztcbiAgICBjYXNlICdkYXknOlxuICAgICAgcmV0dXJuICdEYXlzJztcbiAgfVxufVxuIl19
|
|
@@ -1,15 +1,16 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, ViewChild, } from '@angular/core';
|
|
2
2
|
import { index } from 'd3-array';
|
|
3
3
|
import { scaleBand, scaleLinear } from 'd3-scale';
|
|
4
4
|
import { stack } from 'd3-shape';
|
|
5
|
-
import { timeMonth } from 'd3-time';
|
|
6
5
|
import { timeFormat } from 'd3-time-format';
|
|
7
|
-
import { BehaviorSubject, combineLatest, map, shareReplay } from 'rxjs';
|
|
6
|
+
import { BehaviorSubject, combineLatest, map, shareReplay, } from 'rxjs';
|
|
7
|
+
import { getIntervalTitle, getMajorInterval, getMajorIntervalFormat, getMinorIntervalFormat, getTimeInterval, intervals, } from '../intervals';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
9
9
|
import * as i1 from "@angular/common";
|
|
10
10
|
import * as i2 from "../../modal/callout/callout.component";
|
|
11
11
|
import * as i3 from "../../modal/callout/callout.directive";
|
|
12
|
-
import * as i4 from "
|
|
12
|
+
import * as i4 from "../../input/single-select/single-select.component";
|
|
13
|
+
import * as i5 from "../zero-state/zero-state.component";
|
|
13
14
|
function getDateRange(stacks) {
|
|
14
15
|
let min = Infinity;
|
|
15
16
|
let max = -Infinity;
|
|
@@ -24,6 +25,20 @@ function getDateRange(stacks) {
|
|
|
24
25
|
}
|
|
25
26
|
return [new Date(min), new Date(max)];
|
|
26
27
|
}
|
|
28
|
+
function pointReducer(points) {
|
|
29
|
+
return points.reduce((total, { value }) => total + value, 0);
|
|
30
|
+
}
|
|
31
|
+
const MAX_COLUMNS = 72;
|
|
32
|
+
const DEFAULT_INTERVAL = 'month';
|
|
33
|
+
function getColumnWidth(numColumns) {
|
|
34
|
+
if (numColumns <= 6)
|
|
35
|
+
return 72;
|
|
36
|
+
if (numColumns <= 18)
|
|
37
|
+
return 40;
|
|
38
|
+
if (numColumns <= 40)
|
|
39
|
+
return 16;
|
|
40
|
+
return 8;
|
|
41
|
+
}
|
|
27
42
|
// TODO: once we upgrade to Angular 16, this component can be cleaned up with
|
|
28
43
|
// signals instead of RxJS.
|
|
29
44
|
export class StackedColumnComponent {
|
|
@@ -32,27 +47,91 @@ export class StackedColumnComponent {
|
|
|
32
47
|
this.width$ = new BehaviorSubject(960);
|
|
33
48
|
this.height$ = new BehaviorSubject(256);
|
|
34
49
|
this.valueFormatter = v => v.toString();
|
|
35
|
-
this.
|
|
50
|
+
this.interval$ = new BehaviorSubject(DEFAULT_INTERVAL);
|
|
51
|
+
this.allowedIntervals$ = this.input$.pipe(map(input => {
|
|
52
|
+
const numVisibleColumns = input.filter(stack => stack.filter(s => s.style === 'tooltipOnly').length === 0).length;
|
|
53
|
+
const [minDate, maxDate] = getDateRange(input);
|
|
54
|
+
return intervals.filter(interval => {
|
|
55
|
+
const timeInterval = getTimeInterval(interval);
|
|
56
|
+
const xStepsCount = timeInterval.range(timeInterval.floor(minDate), timeInterval.floor(timeInterval.offset(maxDate, 1))).length;
|
|
57
|
+
return numVisibleColumns * xStepsCount <= MAX_COLUMNS;
|
|
58
|
+
});
|
|
59
|
+
}), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
60
|
+
this.intervalOptions$ = this.allowedIntervals$.pipe(map(intervals => intervals.map((interval) => ({
|
|
61
|
+
id: interval,
|
|
62
|
+
title: getIntervalTitle(interval),
|
|
63
|
+
}))), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
64
|
+
this.selectedIntervalOption$ = combineLatest([
|
|
65
|
+
this.interval$,
|
|
66
|
+
this.intervalOptions$,
|
|
67
|
+
]).pipe(map(([interval, options]) => {
|
|
68
|
+
const allowedIntervals = options.map(({ id }) => id);
|
|
69
|
+
if (!allowedIntervals.includes(interval)) {
|
|
70
|
+
const firstAllowed = allowedIntervals.at(0);
|
|
71
|
+
const lastAllowed = allowedIntervals.at(-1);
|
|
72
|
+
if (!firstAllowed || !lastAllowed)
|
|
73
|
+
return null;
|
|
74
|
+
if (intervals.indexOf(interval) < intervals.indexOf(firstAllowed))
|
|
75
|
+
interval = firstAllowed;
|
|
76
|
+
if (intervals.indexOf(interval) > intervals.indexOf(lastAllowed))
|
|
77
|
+
interval = lastAllowed;
|
|
78
|
+
}
|
|
79
|
+
return options.find(o => o.id === interval) ?? null;
|
|
80
|
+
}), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
81
|
+
this.selectedInterval$ = this.selectedIntervalOption$.pipe(map(option => (option ? option.id : DEFAULT_INTERVAL)));
|
|
82
|
+
this.binnedData$ = combineLatest([
|
|
36
83
|
this.input$,
|
|
84
|
+
this.selectedInterval$,
|
|
85
|
+
]).pipe(map(([input, interval]) => input.map((stack) => {
|
|
86
|
+
const timeInterval = getTimeInterval(interval);
|
|
87
|
+
return stack.map(series => {
|
|
88
|
+
const binned = new Map();
|
|
89
|
+
for (const point of series.data) {
|
|
90
|
+
const bin = timeInterval.floor(point.date).valueOf();
|
|
91
|
+
if (!binned.has(bin)) {
|
|
92
|
+
binned.set(bin, []);
|
|
93
|
+
}
|
|
94
|
+
binned.get(bin).push(point);
|
|
95
|
+
}
|
|
96
|
+
return {
|
|
97
|
+
...series,
|
|
98
|
+
data: [...binned.entries()].map(([dateValue, points]) => ({
|
|
99
|
+
date: new Date(dateValue),
|
|
100
|
+
value: pointReducer(points),
|
|
101
|
+
})),
|
|
102
|
+
};
|
|
103
|
+
});
|
|
104
|
+
})), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
105
|
+
this.drawData$ = combineLatest([
|
|
106
|
+
this.binnedData$,
|
|
37
107
|
this.width$,
|
|
38
108
|
this.height$,
|
|
39
|
-
|
|
109
|
+
this.selectedInterval$,
|
|
110
|
+
]).pipe(map(([binnedData, width, height, interval]) => {
|
|
40
111
|
const viewBox = `0 0 ${width} ${height}`;
|
|
41
|
-
const padding =
|
|
42
|
-
const invisColumns =
|
|
43
|
-
const
|
|
44
|
-
const
|
|
45
|
-
const
|
|
112
|
+
const padding = 16;
|
|
113
|
+
const invisColumns = binnedData.filter(stack => stack.filter(s => s.style == 'tooltipOnly').length > 0).length;
|
|
114
|
+
const visibleColumns = binnedData.length - invisColumns;
|
|
115
|
+
const [minDate, maxDate] = getDateRange(binnedData);
|
|
116
|
+
const hasMultipleYears = minDate.getFullYear() < maxDate.getFullYear();
|
|
117
|
+
const xMinorIntervalFormat = getMinorIntervalFormat(interval);
|
|
118
|
+
const minorFormatter = timeFormat(xMinorIntervalFormat);
|
|
119
|
+
const minorTimeInterval = getTimeInterval(interval);
|
|
120
|
+
const xSteps = minorTimeInterval.range(minorTimeInterval.floor(minDate), minorTimeInterval.floor(minorTimeInterval.offset(maxDate, 1)));
|
|
121
|
+
const xMajorInterval = getMajorInterval(interval);
|
|
122
|
+
const xMajorIntervalFormat = getMajorIntervalFormat(xMajorInterval, hasMultipleYears);
|
|
123
|
+
const majorFormatter = timeFormat(xMajorIntervalFormat);
|
|
124
|
+
const fullFormatter = timeFormat(`${xMinorIntervalFormat} ${xMajorIntervalFormat}`);
|
|
46
125
|
const xOuterScale = scaleBand()
|
|
47
|
-
.domain(xSteps.map(date =>
|
|
126
|
+
.domain(xSteps.map(date => fullFormatter(date)))
|
|
48
127
|
.range([0, width]);
|
|
49
|
-
const columnWidth =
|
|
128
|
+
const columnWidth = getColumnWidth(xSteps.length * visibleColumns);
|
|
50
129
|
const columnPadding = 4;
|
|
51
130
|
const outerPadding = (xOuterScale.bandwidth() -
|
|
52
|
-
(columnPadding * (
|
|
53
|
-
columnWidth * (
|
|
131
|
+
(columnPadding * (binnedData.length - 1 - invisColumns) +
|
|
132
|
+
columnWidth * (binnedData.length - invisColumns))) /
|
|
54
133
|
2;
|
|
55
|
-
const renderInformation =
|
|
134
|
+
const renderInformation = binnedData.map(stackData => {
|
|
56
135
|
const stackedTable = stackData.reduce((table, s) => [
|
|
57
136
|
...table,
|
|
58
137
|
...s.data.map(point => ({
|
|
@@ -94,7 +173,7 @@ export class StackedColumnComponent {
|
|
|
94
173
|
}
|
|
95
174
|
const width = seriesPoint.series.style == 'tooltipOnly' ? 0 : columnWidth;
|
|
96
175
|
const height = yScale(baseline) - yScale(topline);
|
|
97
|
-
const x = (xOuterScale(
|
|
176
|
+
const x = (xOuterScale(fullFormatter(date)) ?? 0) +
|
|
98
177
|
outerPadding +
|
|
99
178
|
columnWidth * (index - invisColumnsSoFar) +
|
|
100
179
|
columnPadding * (index - invisColumnsSoFar);
|
|
@@ -119,17 +198,27 @@ export class StackedColumnComponent {
|
|
|
119
198
|
invisColumnsSoFar++;
|
|
120
199
|
}
|
|
121
200
|
});
|
|
122
|
-
const
|
|
201
|
+
const xMinorTicks = xSteps.map(date => ({
|
|
123
202
|
dateValue: date.valueOf(),
|
|
124
|
-
label:
|
|
125
|
-
x: (xOuterScale(
|
|
203
|
+
label: minorFormatter(date),
|
|
204
|
+
x: (xOuterScale(fullFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,
|
|
126
205
|
}));
|
|
206
|
+
const xMajorTicks = interval !== 'year'
|
|
207
|
+
? xSteps
|
|
208
|
+
.map(date => ({
|
|
209
|
+
dateValue: date.valueOf(),
|
|
210
|
+
label: majorFormatter(date),
|
|
211
|
+
x: (xOuterScale(fullFormatter(date)) ?? 0) +
|
|
212
|
+
xOuterScale.bandwidth() / 2,
|
|
213
|
+
}))
|
|
214
|
+
.filter((tick, i, ticks) => i === 0 || tick.label !== ticks[i - 1].label)
|
|
215
|
+
: [];
|
|
127
216
|
const yTicks = yScale.ticks(5);
|
|
128
217
|
const hoverBandWidth = (renderInformation.length - invisColumns) * columnWidth +
|
|
129
218
|
(renderInformation.length + 1 - invisColumns) * columnPadding;
|
|
130
219
|
const hoverBands = xSteps.map(date => ({
|
|
131
220
|
dateValue: date.valueOf(),
|
|
132
|
-
x: (xOuterScale(
|
|
221
|
+
x: (xOuterScale(fullFormatter(date)) ?? 0) +
|
|
133
222
|
(xOuterScale.bandwidth() - hoverBandWidth) / 2,
|
|
134
223
|
y: yScale(domainMax),
|
|
135
224
|
width: hoverBandWidth,
|
|
@@ -140,7 +229,8 @@ export class StackedColumnComponent {
|
|
|
140
229
|
hoverBands,
|
|
141
230
|
rects,
|
|
142
231
|
viewBox,
|
|
143
|
-
|
|
232
|
+
xMinorTicks,
|
|
233
|
+
xMajorTicks,
|
|
144
234
|
yScale,
|
|
145
235
|
yTicks,
|
|
146
236
|
};
|
|
@@ -181,12 +271,21 @@ export class StackedColumnComponent {
|
|
|
181
271
|
get height() {
|
|
182
272
|
return this.height$.getValue();
|
|
183
273
|
}
|
|
274
|
+
set interval(v) {
|
|
275
|
+
this.interval$.next(v);
|
|
276
|
+
}
|
|
277
|
+
get interval() {
|
|
278
|
+
return this.interval$.getValue();
|
|
279
|
+
}
|
|
280
|
+
setIntervalOption(option) {
|
|
281
|
+
this.interval = option.id;
|
|
282
|
+
}
|
|
184
283
|
}
|
|
185
284
|
StackedColumnComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedColumnComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
186
|
-
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" }, 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.
|
|
285
|
+
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 [options]=\"(intervalOptions$ | async) || []\"\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-0-font-size);line-height:var(--type-0-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: ["options", "selectedOption", "filterabilityOptions", "loading", "noOptionsMessage", "nodeTemplate", "placeholder"], outputs: ["filterQueryChange", "selectedOptionChange"] }, { kind: "component", type: i5.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
187
286
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedColumnComponent, decorators: [{
|
|
188
287
|
type: Component,
|
|
189
|
-
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.
|
|
288
|
+
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 [options]=\"(intervalOptions$ | async) || []\"\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-0-font-size);line-height:var(--type-0-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"] }]
|
|
190
289
|
}], propDecorators: { input: [{
|
|
191
290
|
type: Input
|
|
192
291
|
}], width: [{
|
|
@@ -195,5 +294,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
195
294
|
type: Input
|
|
196
295
|
}], valueFormatter: [{
|
|
197
296
|
type: Input
|
|
297
|
+
}], interval: [{
|
|
298
|
+
type: Input
|
|
299
|
+
}], controls: [{
|
|
300
|
+
type: ViewChild,
|
|
301
|
+
args: ['controls']
|
|
198
302
|
}] } });
|
|
199
|
-
//# 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 { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { index } from 'd3-array';\nimport { scaleBand, scaleLinear } from 'd3-scale';\nimport { stack } from 'd3-shape';\nimport { timeMonth } from 'd3-time';\nimport { timeFormat } from 'd3-time-format';\nimport { BehaviorSubject, combineLatest, map, shareReplay } from 'rxjs';\n\nfunction getDateRange(stacks: Stack[]): [Date, Date] {\n  let min = Infinity;\n  let max = -Infinity;\n  for (const stack of stacks) {\n    for (const series of stack) {\n      for (const { date } of series.data) {\n        const value = date.valueOf();\n        min = Math.min(min, value);\n        max = Math.max(max, value);\n      }\n    }\n  }\n  return [new Date(min), new Date(max)];\n}\n\nexport type Value = { date: Date; value: number };\n\nexport 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\nexport type Stack = Series[];\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<Stack[]>([]);\n  @Input()\n  public set input(v: Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): 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  readonly drawData$ = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      const viewBox = `0 0 ${width} ${height}`;\n      const padding = 8;\n\n      const invisColumns = input.filter(\n        stack => stack.filter(s => s.style == 'tooltipOnly').length > 0,\n      ).length;\n\n      const [minDate, maxDate] = getDateRange(input);\n      const xSteps = timeMonth.range(\n        timeMonth.floor(minDate),\n        timeMonth.floor(timeMonth.offset(maxDate, 1)),\n      );\n      const xFormatter = timeFormat('%b');\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => xFormatter(date)))\n        .range([0, width]);\n\n      const columnWidth = 10;\n      const columnPadding = 4;\n      const outerPadding =\n        (xOuterScale.bandwidth() -\n          (columnPadding * (input.length - 1 - invisColumns) +\n            columnWidth * (input.length - invisColumns))) /\n        2;\n\n      type SeriesPoint = Value & { series: Series };\n\n      const renderInformation = input.map(stackData => {\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        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\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      let domainMax = 0;\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        [...indexedTable.values()].forEach((pointMap, i) => {\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      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      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        let invisColumnFound = false;\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\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(xFormatter(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 xTicks = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        label: xFormatter(date),\n        x: (xOuterScale(xFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,\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(xFormatter(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        xTicks,\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","<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.xTicks\">\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.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"]}
|
|
303
|
+
//# 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 { timeFormat } 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';\n\nfunction getDateRange(stacks: Stack[]): [Date, Date] {\n  let min = Infinity;\n  let max = -Infinity;\n  for (const stack of stacks) {\n    for (const series of stack) {\n      for (const { date } of series.data) {\n        const value = date.valueOf();\n        min = Math.min(min, value);\n        max = Math.max(max, value);\n      }\n    }\n  }\n  return [new Date(min), new Date(max)];\n}\n\nexport type Value = { date: Date; value: number };\n\nexport 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\nexport type Stack = Series[];\n\nfunction pointReducer(points: Value[]) {\n  return points.reduce((total, { value }) => total + value, 0);\n}\n\nconst MAX_COLUMNS = 72;\nconst DEFAULT_INTERVAL: Interval = 'month';\n\nfunction getColumnWidth(numColumns: number) {\n  if (numColumns <= 6) return 72;\n  if (numColumns <= 18) return 40;\n  if (numColumns <= 40) return 16;\n  return 8;\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-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<Stack[]>([]);\n  @Input()\n  public set input(v: Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): 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): Stack => {\n        const timeInterval = getTimeInterval(interval);\n\n        return stack.map(series => {\n          const binned = new Map<number, 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 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 = timeFormat(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 = timeFormat(xMajorIntervalFormat);\n      const fullFormatter = timeFormat(\n        `${xMinorIntervalFormat} ${xMajorIntervalFormat}`,\n      );\n\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => fullFormatter(date)))\n        .range([0, 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 = Value & { series: Series };\n\n      const renderInformation = binnedData.map(stackData => {\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        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\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      let domainMax = 0;\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        [...indexedTable.values()].forEach((pointMap, i) => {\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      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      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        let invisColumnFound = false;\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\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","<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    [options]=\"(intervalOptions$ | async) || []\"\n    [selectedOption]=\"selectedIntervalOption$ | async\"\n    (selectedOptionChange)=\"setIntervalOption($event)\"\n  ></riv-single-select>\n</ng-template>\n"]}
|