@rivet-health/design-system 5.0.2 → 5.2.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 +4 -1
- package/esm2020/lib/visualization/legend-item/legend-item.component.mjs +9 -3
- package/esm2020/lib/visualization/stacked-row/stacked-row.component.mjs +70 -25
- package/esm2020/lib/visualization/time-series/time-series.component.mjs +3 -3
- package/fesm2015/rivet-health-design-system.mjs +82 -28
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +82 -28
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/lib/input/single-select/single-select.component.d.ts +1 -0
- package/lib/visualization/legend-item/legend-item.component.d.ts +3 -1
- package/lib/visualization/stacked-row/stacked-row.component.d.ts +20 -12
- package/package.json +1 -1
|
@@ -43,6 +43,9 @@ export class SingleSelectComponent extends InputLabelComponent {
|
|
|
43
43
|
this.open = true;
|
|
44
44
|
}
|
|
45
45
|
}
|
|
46
|
+
close() {
|
|
47
|
+
this.open = false;
|
|
48
|
+
}
|
|
46
49
|
trackByHeader(i, group) {
|
|
47
50
|
return `${i}${group.header}`;
|
|
48
51
|
}
|
|
@@ -157,4 +160,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
157
160
|
}
|
|
158
161
|
SingleSelectComponent.getFilterAsync = getFilterAsync;
|
|
159
162
|
})(SingleSelectComponent || (SingleSelectComponent = {}));
|
|
160
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/single-select/single-select.component.ts","../../../../../../projects/riv/src/lib/input/single-select/single-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport Fuse from 'fuse.js';\nimport {\n  Observable,\n  debounceTime,\n  from,\n  map,\n  pairwise,\n  startWith,\n  switchMap,\n} from 'rxjs';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\nimport { InputLabelComponent } from '../input-label/input-label.component';\n\n@Component({\n  selector: 'riv-single-select',\n  templateUrl: './single-select.component.html',\n  styleUrls: ['./single-select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SingleSelectComponent<\n  T extends SingleSelectComponent.OptionGroup<U>,\n  U extends SingleSelectComponent.Option,\n> extends InputLabelComponent {\n  @Input()\n  groups: T[] = [];\n\n  @Input()\n  selectedOption: U | null = null;\n\n  @Input()\n  filterabilityOptions: SingleSelectComponent.FilterabilityOptions = {\n    enabled: false,\n  };\n\n  @Input()\n  loading: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @Input()\n  noOptionsMessage: string = 'No available options';\n\n  @Input()\n  nodeTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  triggerTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  placeholder: string = 'Select...';\n\n  @Input()\n  disabled: boolean = false;\n\n  filterQuery: string = '';\n\n  @Output()\n  filterQueryChange = new EventEmitter<string>();\n\n  @Output()\n  selectedOptionChange = new EventEmitter<U>();\n\n  @ViewChild('customTriggerButton') customTriggerButton?: ElementRef;\n  @ViewChild('standardTriggerButton') standardTriggerButton?: ElementRef;\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  getFilterPlaceholder(): string {\n    if (\n      this.filterabilityOptions.enabled &&\n      this.filterabilityOptions.placeholder\n    ) {\n      return this.filterabilityOptions.placeholder;\n    }\n    return 'Filter options...';\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.open = true;\n    }\n  }\n\n  trackByHeader(i: number, group: SingleSelectComponent.OptionGroup<U>) {\n    return `${i}${group.header}`;\n  }\n\n  trackByOption(_: number, option: U) {\n    return option.id;\n  }\n\n  open: boolean = false;\n}\n\nexport namespace SingleSelectComponent {\n  export type Option = {\n    id: string;\n    title: string;\n    subtitle?: string;\n    count?: number;\n    disabled?: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n    subtitleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  type Node<V extends Option> = V & {\n    selected: boolean;\n  };\n\n  export type OptionGroup<O extends Option> = {\n    header?: string;\n    options: O[];\n  };\n\n  export type FilterabilityOptions =\n    | {\n        enabled: false;\n      }\n    | {\n        enabled: true;\n        placeholder?: string;\n      };\n\n  export function getFilterSync<T extends OptionGroup<U>, U extends Option>(\n    groups: T[],\n    exact: boolean = false,\n  ): (query: string) => T[] {\n    return function filter(query: string) {\n      if (!query) return groups;\n\n      const fuseOptions = {\n        keys: ['title', 'subtitle'],\n        includeMatches: true,\n        shouldSort: true,\n        threshold: exact ? 0.0 : 0.6,\n      };\n\n      return groups\n        .map(group => {\n          const fuse = new Fuse(group.options, fuseOptions);\n          const matches = fuse.search(query);\n\n          return {\n            ...group,\n            options: matches.map(match => ({\n              ...match.item,\n              titleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'title')\n                  ?.indices || []),\n              ],\n              subtitleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'subtitle')\n                  ?.indices || []),\n              ],\n            })),\n          };\n        })\n        .filter(group => group.options.length > 0); // Remove empty groups\n    };\n  }\n\n  export function getFilterAsync<T extends OptionGroup<U>, U extends Option>(\n    query$: Observable<string>,\n    fetcher: (query: string) => Promise<T[]>,\n    debounceInput: number = 300,\n  ): Observable<{\n    loading: boolean;\n    groups: T[];\n  }> {\n    return query$.pipe(\n      debounceTime(debounceInput),\n      switchMap(query =>\n        from(fetcher(query)).pipe(\n          map(groups => {\n            if (!query) return { loading: false, groups };\n\n            const fuseOptions = {\n              keys: ['title', 'subtitle'],\n              includeMatches: true,\n              shouldSort: true,\n              threshold: 0.6,\n            };\n\n            const filteredGroups = groups\n              .map(group => {\n                const fuse = new Fuse(group.options, fuseOptions);\n                const matches = fuse.search(query);\n\n                return {\n                  ...group,\n                  options: matches.map(match => ({\n                    ...match.item,\n                    titleHighlightIndices: [\n                      ...(match.matches?.find(result => result.key === 'title')\n                        ?.indices || []),\n                    ],\n                    subtitleHighlightIndices: [\n                      ...(match.matches?.find(\n                        result => result.key === 'subtitle',\n                      )?.indices || []),\n                    ],\n                  })),\n                };\n              })\n              .filter(group => group.options.length > 0); // Remove empty groups\n\n            return { loading: false, groups: filteredGroups };\n          }),\n          startWith({ loading: true, groups: [] }),\n        ),\n      ),\n      pairwise(),\n      map(([previous, next]) => ({\n        loading: next.loading,\n        groups: next.loading ? previous.groups : next.groups,\n      })),\n      startWith({ loading: false, groups: [] }),\n    );\n  }\n}\n","<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n  <button\n    #customTriggerButton\n    (click)=\"allowedOpen()\"\n    [disabled]=\"disabled || locked\"\n    type=\"button\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"triggerTemplate\"\n      [ngTemplateOutletContext]=\"{ selectedOption }\"\n    ></ng-container>\n  </button>\n</ng-container>\n<ng-template #standardTrigger>\n  <riv-input-label\n    [label]=\"label\"\n    [help]=\"help\"\n    [required]=\"required\"\n    [labelActionText]=\"labelActionText\"\n    (labelAction)=\"labelAction.emit($event)\"\n  >\n    <button\n      #standardTriggerButton\n      class=\"trigger\"\n      (click)=\"allowedOpen()\"\n      [disabled]=\"disabled || locked\"\n      type=\"button\"\n    >\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 *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n        <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n      </span>\n    </button>\n  </riv-input-label>\n</ng-template>\n\n<riv-callout\n  *ngIf=\"open\"\n  [anchor]=\"getTrigger()\"\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 *ngFor=\"let group of groups; trackBy: trackByHeader\">\n        <ng-container *ngIf=\"group.header\">\n          <span class=\"group-header\">\n            {{ group.header }}\n          </span>\n        </ng-container>\n        <ng-container *ngIf=\"group.options; let nodes\">\n          <ng-container *ngIf=\"nodes.length > 0; else empty\">\n            <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\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                  type=\"button\"\n                >\n                  <ng-container\n                    [ngTemplateOutlet]=\"nodeTemplate\"\n                    [ngTemplateOutletContext]=\"{ node }\"\n                  ></ng-container>\n                </button>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <button\n                  class=\"single-select-node\"\n                  [class.selected]=\"node?.id === selectedOption?.id\"\n                  [class.disabled]=\"node?.disabled\"\n                  [disabled]=\"node?.disabled\"\n                  (click)=\"selectedOptionChange.emit(node); open = false\"\n                  type=\"button\"\n                >\n                  <riv-icon\n                    [name]=\"'Check'\"\n                    *ngIf=\"node?.id === selectedOption?.id\"\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      </ng-container>\n    </riv-loading-cover>\n  </div>\n  <div class=\"footer\">\n    <ng-content select=\"[footer]\"> </ng-content>\n  </div>\n</riv-callout>\n"]}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-select.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/single-select/single-select.component.ts","../../../../../../projects/riv/src/lib/input/single-select/single-select.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EAET,YAAY,EACZ,KAAK,EACL,MAAM,EAEN,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,IAAI,MAAM,SAAS,CAAC;AAC3B,OAAO,EAEL,YAAY,EACZ,IAAI,EACJ,GAAG,EACH,QAAQ,EACR,SAAS,EACT,SAAS,GACV,MAAM,MAAM,CAAC;AAEd,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;;;;;;;;AAQ3E,MAAM,OAAO,qBAGX,SAAQ,mBAAmB;IAT7B;;QAWE,WAAM,GAAQ,EAAE,CAAC;QAGjB,mBAAc,GAAa,IAAI,CAAC;QAGhC,yBAAoB,GAA+C;YACjE,OAAO,EAAE,KAAK;SACf,CAAC;QAGF,YAAO,GAAY,KAAK,CAAC;QAGzB,WAAM,GAAY,KAAK,CAAC;QAGxB,qBAAgB,GAAW,sBAAsB,CAAC;QASlD,gBAAW,GAAW,WAAW,CAAC;QAGlC,aAAQ,GAAY,KAAK,CAAC;QAE1B,gBAAW,GAAW,EAAE,CAAC;QAGzB,sBAAiB,GAAG,IAAI,YAAY,EAAU,CAAC;QAG/C,yBAAoB,GAAG,IAAI,YAAY,EAAK,CAAC;QAwC7C,SAAI,GAAY,KAAK,CAAC;KACvB;IApCC,UAAU;QACR,OAAO,CACL,IAAI,CAAC,mBAAmB,EAAE,aAAa;YACvC,IAAI,CAAC,qBAAqB,EAAE,aAAa,CAC1C,CAAC;IACJ,CAAC;IAED,oBAAoB;QAClB,IACE,IAAI,CAAC,oBAAoB,CAAC,OAAO;YACjC,IAAI,CAAC,oBAAoB,CAAC,WAAW,EACrC;YACA,OAAO,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC;SAC9C;QACD,OAAO,mBAAmB,CAAC;IAC7B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAClC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;SAClB;IACH,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;IACpB,CAAC;IAED,aAAa,CAAC,CAAS,EAAE,KAA2C;QAClE,OAAO,GAAG,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC;IAC/B,CAAC;IAED,aAAa,CAAC,CAAS,EAAE,MAAS;QAChC,OAAO,MAAM,CAAC,EAAE,CAAC;IACnB,CAAC;;kHAhFU,qBAAqB;sGAArB,qBAAqB,8sBC7BlC,wmJAsIA;2FDzGa,qBAAqB;kBANjC,SAAS;+BACE,mBAAmB,mBAGZ,uBAAuB,CAAC,MAAM;8BAO/C,MAAM;sBADL,KAAK;gBAIN,cAAc;sBADb,KAAK;gBAIN,oBAAoB;sBADnB,KAAK;gBAMN,OAAO;sBADN,KAAK;gBAIN,MAAM;sBADL,KAAK;gBAIN,gBAAgB;sBADf,KAAK;gBAIN,YAAY;sBADX,KAAK;gBAIN,eAAe;sBADd,KAAK;gBAIN,WAAW;sBADV,KAAK;gBAIN,QAAQ;sBADP,KAAK;gBAMN,iBAAiB;sBADhB,MAAM;gBAIP,oBAAoB;sBADnB,MAAM;gBAG2B,mBAAmB;sBAApD,SAAS;uBAAC,qBAAqB;gBACI,qBAAqB;sBAAxD,SAAS;uBAAC,uBAAuB;;AAwCpC,WAAiB,qBAAqB;IA6BpC,SAAgB,aAAa,CAC3B,MAAW,EACX,QAAiB,KAAK;QAEtB,OAAO,SAAS,MAAM,CAAC,KAAa;YAClC,IAAI,CAAC,KAAK;gBAAE,OAAO,MAAM,CAAC;YAE1B,MAAM,WAAW,GAAG;gBAClB,IAAI,EAAE,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC3B,cAAc,EAAE,IAAI;gBACpB,UAAU,EAAE,IAAI;gBAChB,SAAS,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG;aAC7B,CAAC;YAEF,OAAO,MAAM;iBACV,GAAG,CAAC,KAAK,CAAC,EAAE;gBACX,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC;gBAClD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAEnC,OAAO;oBACL,GAAG,KAAK;oBACR,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;wBAC7B,GAAG,KAAK,CAAC,IAAI;wBACb,qBAAqB,EAAE;4BACrB,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,KAAK,OAAO,CAAC;gCACvD,EAAE,OAAO,IAAI,EAAE,CAAC;yBACnB;wBACD,wBAAwB,EAAE;4BACxB,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,KAAK,UAAU,CAAC;gCAC1D,EAAE,OAAO,IAAI,EAAE,CAAC;yBACnB;qBACF,CAAC,CAAC;iBACJ,CAAC;YACJ,CAAC,CAAC;iBACD,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,sBAAsB;QACtE,CAAC,CAAC;IACJ,CAAC;IApCe,mCAAa,gBAoC5B,CAAA;IAED,SAAgB,cAAc,CAC5B,MAA0B,EAC1B,OAAwC,EACxC,gBAAwB,GAAG;QAK3B,OAAO,MAAM,CAAC,IAAI,CAChB,YAAY,CAAC,aAAa,CAAC,EAC3B,SAAS,CAAC,KAAK,CAAC,EAAE,CAChB,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CACvB,GAAG,CAAC,MAAM,CAAC,EAAE;YACX,IAAI,CAAC,KAAK;gBAAE,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC;YAE9C,MAAM,WAAW,GAAG;gBAClB,IAAI,EAAE,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC3B,cAAc,EAAE,IAAI;gBACpB,UAAU,EAAE,IAAI;gBAChB,SAAS,EAAE,GAAG;aACf,CAAC;YAEF,MAAM,cAAc,GAAG,MAAM;iBAC1B,GAAG,CAAC,KAAK,CAAC,EAAE;gBACX,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC;gBAClD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAEnC,OAAO;oBACL,GAAG,KAAK;oBACR,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;wBAC7B,GAAG,KAAK,CAAC,IAAI;wBACb,qBAAqB,EAAE;4BACrB,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,KAAK,OAAO,CAAC;gCACvD,EAAE,OAAO,IAAI,EAAE,CAAC;yBACnB;wBACD,wBAAwB,EAAE;4BACxB,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CACrB,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,KAAK,UAAU,CACpC,EAAE,OAAO,IAAI,EAAE,CAAC;yBAClB;qBACF,CAAC,CAAC;iBACJ,CAAC;YACJ,CAAC,CAAC;iBACD,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,sBAAsB;YAEpE,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,cAAc,EAAE,CAAC;QACpD,CAAC,CAAC,EACF,SAAS,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,CACzC,CACF,EACD,QAAQ,EAAE,EACV,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;YACzB,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM;SACrD,CAAC,CAAC,EACH,SAAS,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,CAC1C,CAAC;IACJ,CAAC;IAzDe,oCAAc,iBAyD7B,CAAA;AACH,CAAC,EA7HgB,qBAAqB,KAArB,qBAAqB,QA6HrC","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport Fuse from 'fuse.js';\nimport {\n  Observable,\n  debounceTime,\n  from,\n  map,\n  pairwise,\n  startWith,\n  switchMap,\n} from 'rxjs';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\nimport { InputLabelComponent } from '../input-label/input-label.component';\n\n@Component({\n  selector: 'riv-single-select',\n  templateUrl: './single-select.component.html',\n  styleUrls: ['./single-select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SingleSelectComponent<\n  T extends SingleSelectComponent.OptionGroup<U>,\n  U extends SingleSelectComponent.Option,\n> extends InputLabelComponent {\n  @Input()\n  groups: T[] = [];\n\n  @Input()\n  selectedOption: U | null = null;\n\n  @Input()\n  filterabilityOptions: SingleSelectComponent.FilterabilityOptions = {\n    enabled: false,\n  };\n\n  @Input()\n  loading: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @Input()\n  noOptionsMessage: string = 'No available options';\n\n  @Input()\n  nodeTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  triggerTemplate?: TemplateRef<unknown>;\n\n  @Input()\n  placeholder: string = 'Select...';\n\n  @Input()\n  disabled: boolean = false;\n\n  filterQuery: string = '';\n\n  @Output()\n  filterQueryChange = new EventEmitter<string>();\n\n  @Output()\n  selectedOptionChange = new EventEmitter<U>();\n\n  @ViewChild('customTriggerButton') customTriggerButton?: ElementRef;\n  @ViewChild('standardTriggerButton') standardTriggerButton?: ElementRef;\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  getFilterPlaceholder(): string {\n    if (\n      this.filterabilityOptions.enabled &&\n      this.filterabilityOptions.placeholder\n    ) {\n      return this.filterabilityOptions.placeholder;\n    }\n    return 'Filter options...';\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.open = true;\n    }\n  }\n\n  public close(): void {\n    this.open = false;\n  }\n\n  trackByHeader(i: number, group: SingleSelectComponent.OptionGroup<U>) {\n    return `${i}${group.header}`;\n  }\n\n  trackByOption(_: number, option: U) {\n    return option.id;\n  }\n\n  open: boolean = false;\n}\n\nexport namespace SingleSelectComponent {\n  export type Option = {\n    id: string;\n    title: string;\n    subtitle?: string;\n    count?: number;\n    disabled?: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n    subtitleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  type Node<V extends Option> = V & {\n    selected: boolean;\n  };\n\n  export type OptionGroup<O extends Option> = {\n    header?: string;\n    options: O[];\n  };\n\n  export type FilterabilityOptions =\n    | {\n        enabled: false;\n      }\n    | {\n        enabled: true;\n        placeholder?: string;\n      };\n\n  export function getFilterSync<T extends OptionGroup<U>, U extends Option>(\n    groups: T[],\n    exact: boolean = false,\n  ): (query: string) => T[] {\n    return function filter(query: string) {\n      if (!query) return groups;\n\n      const fuseOptions = {\n        keys: ['title', 'subtitle'],\n        includeMatches: true,\n        shouldSort: true,\n        threshold: exact ? 0.0 : 0.6,\n      };\n\n      return groups\n        .map(group => {\n          const fuse = new Fuse(group.options, fuseOptions);\n          const matches = fuse.search(query);\n\n          return {\n            ...group,\n            options: matches.map(match => ({\n              ...match.item,\n              titleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'title')\n                  ?.indices || []),\n              ],\n              subtitleHighlightIndices: [\n                ...(match.matches?.find(result => result.key === 'subtitle')\n                  ?.indices || []),\n              ],\n            })),\n          };\n        })\n        .filter(group => group.options.length > 0); // Remove empty groups\n    };\n  }\n\n  export function getFilterAsync<T extends OptionGroup<U>, U extends Option>(\n    query$: Observable<string>,\n    fetcher: (query: string) => Promise<T[]>,\n    debounceInput: number = 300,\n  ): Observable<{\n    loading: boolean;\n    groups: T[];\n  }> {\n    return query$.pipe(\n      debounceTime(debounceInput),\n      switchMap(query =>\n        from(fetcher(query)).pipe(\n          map(groups => {\n            if (!query) return { loading: false, groups };\n\n            const fuseOptions = {\n              keys: ['title', 'subtitle'],\n              includeMatches: true,\n              shouldSort: true,\n              threshold: 0.6,\n            };\n\n            const filteredGroups = groups\n              .map(group => {\n                const fuse = new Fuse(group.options, fuseOptions);\n                const matches = fuse.search(query);\n\n                return {\n                  ...group,\n                  options: matches.map(match => ({\n                    ...match.item,\n                    titleHighlightIndices: [\n                      ...(match.matches?.find(result => result.key === 'title')\n                        ?.indices || []),\n                    ],\n                    subtitleHighlightIndices: [\n                      ...(match.matches?.find(\n                        result => result.key === 'subtitle',\n                      )?.indices || []),\n                    ],\n                  })),\n                };\n              })\n              .filter(group => group.options.length > 0); // Remove empty groups\n\n            return { loading: false, groups: filteredGroups };\n          }),\n          startWith({ loading: true, groups: [] }),\n        ),\n      ),\n      pairwise(),\n      map(([previous, next]) => ({\n        loading: next.loading,\n        groups: next.loading ? previous.groups : next.groups,\n      })),\n      startWith({ loading: false, groups: [] }),\n    );\n  }\n}\n","<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n  <button\n    #customTriggerButton\n    (click)=\"allowedOpen()\"\n    [disabled]=\"disabled || locked\"\n    type=\"button\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"triggerTemplate\"\n      [ngTemplateOutletContext]=\"{ selectedOption }\"\n    ></ng-container>\n  </button>\n</ng-container>\n<ng-template #standardTrigger>\n  <riv-input-label\n    [label]=\"label\"\n    [help]=\"help\"\n    [required]=\"required\"\n    [labelActionText]=\"labelActionText\"\n    (labelAction)=\"labelAction.emit($event)\"\n  >\n    <button\n      #standardTriggerButton\n      class=\"trigger\"\n      (click)=\"allowedOpen()\"\n      [disabled]=\"disabled || locked\"\n      type=\"button\"\n    >\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 *ngIf=\"!locked\" [name]=\"'ChevronDown'\" [size]=\"16\"></riv-icon>\n        <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"16\"></riv-icon>\n      </span>\n    </button>\n  </riv-input-label>\n</ng-template>\n\n<riv-callout\n  *ngIf=\"open\"\n  [anchor]=\"getTrigger()\"\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 *ngFor=\"let group of groups; trackBy: trackByHeader\">\n        <ng-container *ngIf=\"group.header\">\n          <span class=\"group-header\">\n            {{ group.header }}\n          </span>\n        </ng-container>\n        <ng-container *ngIf=\"group.options; let nodes\">\n          <ng-container *ngIf=\"nodes.length > 0; else empty\">\n            <ng-container *ngFor=\"let node of nodes; trackBy: trackByOption\">\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                  type=\"button\"\n                >\n                  <ng-container\n                    [ngTemplateOutlet]=\"nodeTemplate\"\n                    [ngTemplateOutletContext]=\"{ node }\"\n                  ></ng-container>\n                </button>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <button\n                  class=\"single-select-node\"\n                  [class.selected]=\"node?.id === selectedOption?.id\"\n                  [class.disabled]=\"node?.disabled\"\n                  [disabled]=\"node?.disabled\"\n                  (click)=\"selectedOptionChange.emit(node); open = false\"\n                  type=\"button\"\n                >\n                  <riv-icon\n                    [name]=\"'Check'\"\n                    *ngIf=\"node?.id === selectedOption?.id\"\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      </ng-container>\n    </riv-loading-cover>\n  </div>\n  <div class=\"footer\">\n    <ng-content select=\"[footer]\"> </ng-content>\n  </div>\n</riv-callout>\n"]}
|
|
@@ -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
|
|
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
|
|
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,
|
|
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(
|
|
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,42 @@ 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:
|
|
36
|
-
|
|
37
|
-
|
|
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; // calculate base row height
|
|
47
|
+
// Clamping the row height between the min and max heights
|
|
48
|
+
return Math.min(Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
|
|
49
|
+
}
|
|
50
|
+
return (this.RECTANGLE_HEIGHTS.MIN_HEIGHT + this.SECTION_PADDING.BETWEEN_ROW); // Default if no input
|
|
51
|
+
},
|
|
52
|
+
// Max Height of each rectangle
|
|
53
|
+
PER_RECTANGLE: () => {
|
|
54
|
+
//Only use height if specified
|
|
55
|
+
if (this.input.length && this.height) {
|
|
56
|
+
let rectangleHeight = (this.height -
|
|
57
|
+
this.SECTION_HEIGHTS.X_AXIS_LABEL -
|
|
58
|
+
this.SECTION_PADDING.OUTER_TOP_BOTTOM -
|
|
59
|
+
this.SECTION_PADDING.RECTANGLE_TO_ROW) /
|
|
60
|
+
this.input.length -
|
|
61
|
+
this.SECTION_PADDING.BETWEEN_ROW; // calculate base rectangle height
|
|
62
|
+
// Clamping the rectangle height between the min and max heights
|
|
63
|
+
return Math.min(Math.max(rectangleHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
|
|
64
|
+
}
|
|
65
|
+
return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input
|
|
66
|
+
},
|
|
38
67
|
};
|
|
39
68
|
this.SECTION_WIDTHS = {
|
|
40
69
|
// Space reserved for the name at the start of the row
|
|
@@ -52,33 +81,43 @@ export class StackedRowComponent {
|
|
|
52
81
|
RECTANGLE_TO_ROW: 6,
|
|
53
82
|
};
|
|
54
83
|
// The number of X-axis segments to create, with labels at the start of each segment.
|
|
55
|
-
this.
|
|
84
|
+
this.X_TICK_COUNT_MAX = 8;
|
|
85
|
+
this.X_TICK_COUNT_MIN = 3;
|
|
86
|
+
this.X_TICK_MAX_WIDTH = 900;
|
|
87
|
+
this.X_TICK_MIN_WIDTH = 400;
|
|
56
88
|
this.drawData$ = combineLatest([
|
|
57
89
|
this.input$,
|
|
58
90
|
this.width$,
|
|
59
91
|
this.height$,
|
|
60
92
|
]).pipe(map(([input, width, height]) => {
|
|
61
|
-
return this.getDrawnData(input, width, height
|
|
93
|
+
return this.getDrawnData(input, width, height
|
|
94
|
+
? height
|
|
95
|
+
: this.SECTION_HEIGHTS.X_AXIS_LABEL +
|
|
96
|
+
this.SECTION_PADDING.OUTER_TOP_BOTTOM +
|
|
97
|
+
this.SECTION_PADDING.RECTANGLE_TO_ROW +
|
|
98
|
+
this.SECTION_HEIGHTS.PER_ROW() * input.length);
|
|
62
99
|
}), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
63
100
|
// Lets us know whether to display the zero data state
|
|
64
101
|
this.empty$ = this.input$.pipe(map(input => input.every(stack => stack.data.length === 0)), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
65
102
|
// What rectangle we are currently hovering over
|
|
66
103
|
this.hoveredRectangle$ = new BehaviorSubject(null);
|
|
104
|
+
// What band we are currently hovering over
|
|
105
|
+
this.hoveredBand$ = new BehaviorSubject(null);
|
|
67
106
|
// Create the metric to display based on the currently hovered rectangle
|
|
68
|
-
this.callout$ = this.
|
|
69
|
-
if (!
|
|
107
|
+
this.callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(map(([hoveredBand, drawData]) => {
|
|
108
|
+
if (!hoveredBand || !drawData)
|
|
70
109
|
return null;
|
|
71
|
-
const { target } =
|
|
110
|
+
const { target } = hoveredBand.event;
|
|
72
111
|
if (!target)
|
|
73
112
|
return null;
|
|
74
113
|
const anchor = target.getBoundingClientRect();
|
|
75
|
-
const
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
value: this.valueFormatter(stackedRowElement.value),
|
|
80
|
-
};
|
|
81
|
-
return { anchor,
|
|
114
|
+
const metrics = drawData.rects
|
|
115
|
+
.filter(rect => rect.rowName === hoveredBand.rowName)
|
|
116
|
+
.map(rect => ({
|
|
117
|
+
label: rect.stackedRowElement.label,
|
|
118
|
+
value: this.valueFormatter(rect.stackedRowElement.value),
|
|
119
|
+
}));
|
|
120
|
+
return { anchor, metrics };
|
|
82
121
|
}), shareReplay({ refCount: true, bufferSize: 1 }));
|
|
83
122
|
}
|
|
84
123
|
set input(v) {
|
|
@@ -122,7 +161,7 @@ export class StackedRowComponent {
|
|
|
122
161
|
// The y-index of the top of the current row
|
|
123
162
|
getTopOfRow(rowIndex) {
|
|
124
163
|
const spaceAboveRows = this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL;
|
|
125
|
-
const aboveRows = rowIndex * this.SECTION_HEIGHTS.PER_ROW;
|
|
164
|
+
const aboveRows = rowIndex * this.SECTION_HEIGHTS.PER_ROW();
|
|
126
165
|
const aboveRowPaddings = rowIndex * this.SECTION_PADDING.BETWEEN_ROW;
|
|
127
166
|
const allAboveRows = aboveRows + aboveRowPaddings;
|
|
128
167
|
return spaceAboveRows + allAboveRows;
|
|
@@ -143,14 +182,20 @@ export class StackedRowComponent {
|
|
|
143
182
|
const xScale = scaleLinear()
|
|
144
183
|
.domain([0, domainMax])
|
|
145
184
|
.range([this.locationOfXAxisZero(), this.locationOfXAxisEnd(width)]);
|
|
146
|
-
|
|
185
|
+
// Setup a linear scale for tick count based on width
|
|
186
|
+
const tickScale = scaleLinear()
|
|
187
|
+
.domain([this.X_TICK_MIN_WIDTH, this.X_TICK_MAX_WIDTH])
|
|
188
|
+
.range([this.X_TICK_COUNT_MIN, this.X_TICK_COUNT_MAX])
|
|
189
|
+
.clamp(true); // Ensures the scale does not go beyond the defined range
|
|
190
|
+
const xtickCountAdjusted = Math.round(tickScale(width));
|
|
191
|
+
const xTicks = xScale.ticks(xtickCountAdjusted);
|
|
147
192
|
const rects = this.createRectanglesFromRows(rowsForDisplay, xScale);
|
|
148
193
|
const rowNamesToDraw = this.getRowNamesToDraw(rowsForDisplay);
|
|
149
194
|
const rowTotalsToDraw = this.getRowTotalsToDraw(rowsForDisplay, width);
|
|
150
195
|
const legendItems = getLegendItems(rowsForDisplay, keysInOrder);
|
|
151
196
|
return {
|
|
152
197
|
legendItems: legendItems,
|
|
153
|
-
rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE,
|
|
198
|
+
rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE(),
|
|
154
199
|
rects: rects,
|
|
155
200
|
rowNames: rowNamesToDraw,
|
|
156
201
|
rowTotals: rowTotalsToDraw,
|
|
@@ -166,7 +211,7 @@ export class StackedRowComponent {
|
|
|
166
211
|
currentRow.elementsButStacked.forEach((currentElement, _) => {
|
|
167
212
|
const topOfRow = this.getTopOfRow(rowIndex);
|
|
168
213
|
const rectangleWidth = xScale(currentElement.topline) - xScale(currentElement.baseline);
|
|
169
|
-
const rectangleHeight = this.SECTION_HEIGHTS.PER_RECTANGLE;
|
|
214
|
+
const rectangleHeight = this.SECTION_HEIGHTS.PER_RECTANGLE();
|
|
170
215
|
const rectangleX = xScale(currentElement.baseline);
|
|
171
216
|
const rectangleY = topOfRow + this.SECTION_PADDING.RECTANGLE_TO_ROW;
|
|
172
217
|
const fill = `var(${currentElement.colorToken})`;
|
|
@@ -191,7 +236,7 @@ export class StackedRowComponent {
|
|
|
191
236
|
// The row name looks right if its base is at the base of the associated rectangle
|
|
192
237
|
const rowLabelY = this.getTopOfRow(rowIndex) +
|
|
193
238
|
this.SECTION_PADDING.RECTANGLE_TO_ROW +
|
|
194
|
-
this.SECTION_HEIGHTS.PER_RECTANGLE;
|
|
239
|
+
this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;
|
|
195
240
|
return {
|
|
196
241
|
label: row.rowName,
|
|
197
242
|
x: rowLabelX,
|
|
@@ -206,7 +251,7 @@ export class StackedRowComponent {
|
|
|
206
251
|
// The row totals looks right if its base is at the base of the associated rectangle
|
|
207
252
|
const rowLabelY = this.getTopOfRow(rowIndex) +
|
|
208
253
|
this.SECTION_PADDING.RECTANGLE_TO_ROW +
|
|
209
|
-
this.SECTION_HEIGHTS.PER_RECTANGLE;
|
|
254
|
+
this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;
|
|
210
255
|
return {
|
|
211
256
|
label: this.valueFormatter(row.rowTotal),
|
|
212
257
|
x: rowLabelX,
|
|
@@ -216,10 +261,10 @@ export class StackedRowComponent {
|
|
|
216
261
|
}
|
|
217
262
|
}
|
|
218
263
|
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
|
|
264
|
+
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
265
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: StackedRowComponent, decorators: [{
|
|
221
266
|
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
|
|
267
|
+
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
268
|
}], propDecorators: { input: [{
|
|
224
269
|
type: Input
|
|
225
270
|
}], width: [{
|
|
@@ -231,4 +276,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
231
276
|
}], zeroStateMessage: [{
|
|
232
277
|
type: Input
|
|
233
278
|
}] } });
|
|
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"]}
|
|
279
|
+
//# 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; // 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 (\n        this.RECTANGLE_HEIGHTS.MIN_HEIGHT + this.SECTION_PADDING.BETWEEN_ROW\n      ); // 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() * 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"]}
|