@rivet-health/design-system 2.8.0 → 2.8.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/visualization/stacked-column/stacked-column.component.mjs +2 -2
- package/fesm2015/rivet-health-design-system.mjs +1 -1
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +1 -1
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -45,7 +45,7 @@ export class StackedColumnComponent {
|
|
|
45
45
|
const xOuterScale = scaleBand()
|
|
46
46
|
.domain(xSteps.map(date => xFormatter(date)))
|
|
47
47
|
.range([0, width]);
|
|
48
|
-
const columnWidth =
|
|
48
|
+
const columnWidth = 10;
|
|
49
49
|
const columnPadding = 4;
|
|
50
50
|
const outerPadding = (xOuterScale.bandwidth() -
|
|
51
51
|
(columnPadding * (input.length - 1) + columnWidth * input.length)) /
|
|
@@ -180,4 +180,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
180
180
|
}], valueFormatter: [{
|
|
181
181
|
type: Input
|
|
182
182
|
}] } });
|
|
183
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-column.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { index } from 'd3-array';\nimport { scaleBand, scaleLinear } from 'd3-scale';\nimport { stack } from 'd3-shape';\nimport { timeMonth } from 'd3-time';\nimport { timeFormat } from 'd3-time-format';\nimport { BehaviorSubject, combineLatest, map, shareReplay } from 'rxjs';\n\nfunction getDateRange(stacks: Stack[]): [Date, Date] {\n  let min = Infinity;\n  let max = -Infinity;\n  for (const stack of stacks) {\n    for (const series of stack) {\n      for (const { date } of series.data) {\n        const value = date.valueOf();\n        min = Math.min(min, value);\n        max = Math.max(max, value);\n      }\n    }\n  }\n  return [new Date(min), new Date(max)];\n}\n\nexport type Value = { date: Date; value: number };\n\nexport type Series = {\n  label: string;\n  colorToken: string;\n  data: Value[];\n  style?: 'solid' | 'striped';\n};\n\nexport type Stack = Series[];\n\n// TODO: once we upgrade to Angular 16, this component can be cleaned up with\n// signals instead of RxJS.\n@Component({\n  selector: 'riv-stacked-column',\n  templateUrl: './stacked-column.component.html',\n  styleUrls: ['./stacked-column.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedColumnComponent {\n  private readonly input$ = new BehaviorSubject<Stack[]>([]);\n  @Input()\n  public set input(v: Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): Stack[] {\n    return this.input$.getValue();\n  }\n\n  private readonly width$ = new BehaviorSubject<number>(960);\n  @Input()\n  public set width(v: number) {\n    this.width$.next(v);\n  }\n  public get width(): number {\n    return this.width$.getValue();\n  }\n\n  private readonly height$ = new BehaviorSubject<number>(256);\n  @Input()\n  public set height(v: number) {\n    this.height$.next(v);\n  }\n  public get height(): number {\n    return this.height$.getValue();\n  }\n\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  readonly drawData$ = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      const viewBox = `0 0 ${width} ${height}`;\n      const padding = 8;\n\n      const [minDate, maxDate] = getDateRange(input);\n      const xSteps = timeMonth.range(\n        timeMonth.floor(minDate),\n        timeMonth.floor(timeMonth.offset(maxDate, 1)),\n      );\n      const xFormatter = timeFormat('%b');\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => xFormatter(date)))\n        .range([0, width]);\n\n      const columnWidth = 6;\n      const columnPadding = 4;\n      const outerPadding =\n        (xOuterScale.bandwidth() -\n          (columnPadding * (input.length - 1) + columnWidth * input.length)) /\n        2;\n\n      type SeriesPoint = Value & { series: Series };\n\n      const renderInformation = input.map(stackData => {\n        const stackedTable = stackData.reduce<SeriesPoint[]>(\n          (table, s) => [\n            ...table,\n            ...s.data.map(point => ({\n              ...point,\n              series: s,\n            })),\n          ],\n          [],\n        );\n        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\n\n        const stacked = stack<\n          typeof indexedTable extends Map<infer K, infer V> ? [K, V] : never\n        >()\n          .keys(stackData.map(v => v.label))\n          .value(([_, group], key) => {\n            return group.get(key)!.value;\n          })(indexedTable);\n\n        return { indexedTable, stacked };\n      });\n\n      let domainMax = 0;\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        [...indexedTable.values()].forEach((pointMap, i) => {\n          [...pointMap.values()].forEach((_, j) => {\n            domainMax = Math.max(domainMax, ...stacked[j][i]);\n          });\n        });\n      });\n      const yScale = scaleLinear()\n        .domain([0, domainMax])\n        .range([height - padding * 2, padding]);\n\n      const rects: {\n        dateValue: number;\n        x: number;\n        y: number;\n        width: number;\n        height: number;\n        fill: string;\n        striped: boolean;\n        seriesPoint: SeriesPoint;\n      }[] = [];\n      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\n            const [baseline, topline] = stacked[j][i];\n\n            const width = columnWidth;\n            const height = yScale(baseline) - yScale(topline);\n            const x =\n              (xOuterScale(xFormatter(date)) ?? 0) +\n              outerPadding +\n              columnWidth * index +\n              columnPadding * index;\n            const y = yScale(baseline) - height;\n            const fill = `var(${seriesPoint.series.colorToken})`;\n\n            rects.push({\n              dateValue: date.valueOf(),\n              x,\n              y,\n              width,\n              height,\n              fill,\n              striped: seriesPoint.series.style === 'striped',\n              seriesPoint,\n            });\n          });\n        });\n      });\n\n      const xTicks = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        label: xFormatter(date),\n        x: (xOuterScale(xFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,\n      }));\n      const yTicks = yScale.ticks(5);\n\n      const hoverBandWidth =\n        renderInformation.length * columnWidth +\n        (renderInformation.length + 1) * columnPadding;\n      const hoverBands = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        x:\n          (xOuterScale(xFormatter(date)) ?? 0) +\n          (xOuterScale.bandwidth() - hoverBandWidth) / 2,\n        y: yScale(domainMax),\n        width: hoverBandWidth,\n        height: yScale(0) - yScale(domainMax),\n      }));\n\n      return {\n        columnWidth,\n        hoverBands,\n        rects,\n        viewBox,\n        xTicks,\n        yScale,\n        yTicks,\n      };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly empty$ = this.input$.pipe(\n    map(input =>\n      input.every(stack => stack.every(series => series.data.length === 0)),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly hoveredBand$ = new BehaviorSubject<{\n    dateValue: number;\n    event: MouseEvent;\n  } | null>(null);\n\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand) return null;\n\n      const { target } = hoveredBand.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      type Metric = { label: string; value: string };\n      const metrics: Metric[] = drawData.rects\n        .filter(rect => rect.dateValue === hoveredBand.dateValue)\n        .map(rect => ({\n          label: rect.seriesPoint.series.label,\n          value: this.valueFormatter(rect.seriesPoint.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n","<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n  <svg\n    xmlns=\"http://www.w3.org/2000/svg\"\n    *ngIf=\"drawData$ | async; let d\"\n    [attr.viewBox]=\"d.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"d.columnWidth\"\n        [attr.height]=\"d.columnWidth\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          x1=\"0\"\n          [attr.y1]=\"d.columnWidth\"\n          [attr.x2]=\"d.columnWidth\"\n          y2=\"0\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let tick of d.xTicks\">\n      <rect\n        class=\"tick-background\"\n        [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n        [attr.x]=\"tick.x - 20\"\n        [attr.y]=\"d.yScale(0) + 3\"\n        width=\"40\"\n        height=\"12\"\n        rx=\"2\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 12\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.yTicks\">\n      <rect\n        class=\"tick\"\n        x=\"0\"\n        [attr.y]=\"d.yScale(tick)\"\n        width=\"100%\"\n        height=\"1\"\n      ></rect>\n      <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g *ngFor=\"let rect of d.rects\">\n      <rect\n        class=\"column\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        [attr.fill]=\"rect.fill\"\n        [class.blurred]=\"\n          (hoveredBand$ | async) !== null &&\n          (hoveredBand$ | async)?.dateValue !== rect.dateValue\n        \"\n      ></rect>\n      <rect\n        *ngIf=\"rect.striped\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        fill=\"url(#stripes)\"\n      ></rect>\n    </g>\n    <rect\n      *ngFor=\"let rect of d.hoverBands\"\n      [attr.x]=\"rect.x\"\n      [attr.y]=\"rect.y\"\n      [attr.width]=\"rect.width\"\n      [attr.height]=\"rect.height\"\n      fill=\"transparent\"\n      (mouseenter)=\"\n        hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n      \"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    ></rect>\n  </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n  <riv-callout\n    *riv-callout\n    [anchor]=\"callout.anchor\"\n    [isModal]=\"false\"\n    [preferredPosition]=\"'center-right'\"\n    [allowedPositions]=\"[\n      'center-right',\n      'center-left',\n      'top-center',\n      'bottom-center'\n    ]\"\n  >\n    <div class=\"callout-content\">\n      <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n        <div>{{ metric.label }}</div>\n        <div class=\"callout-metric-value\">{{ metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state></riv-zero-state>\n</ng-template>\n"]}
|
|
183
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-column.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-column/stacked-column.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { index } from 'd3-array';\nimport { scaleBand, scaleLinear } from 'd3-scale';\nimport { stack } from 'd3-shape';\nimport { timeMonth } from 'd3-time';\nimport { timeFormat } from 'd3-time-format';\nimport { BehaviorSubject, combineLatest, map, shareReplay } from 'rxjs';\n\nfunction getDateRange(stacks: Stack[]): [Date, Date] {\n  let min = Infinity;\n  let max = -Infinity;\n  for (const stack of stacks) {\n    for (const series of stack) {\n      for (const { date } of series.data) {\n        const value = date.valueOf();\n        min = Math.min(min, value);\n        max = Math.max(max, value);\n      }\n    }\n  }\n  return [new Date(min), new Date(max)];\n}\n\nexport type Value = { date: Date; value: number };\n\nexport type Series = {\n  label: string;\n  colorToken: string;\n  data: Value[];\n  style?: 'solid' | 'striped';\n};\n\nexport type Stack = Series[];\n\n// TODO: once we upgrade to Angular 16, this component can be cleaned up with\n// signals instead of RxJS.\n@Component({\n  selector: 'riv-stacked-column',\n  templateUrl: './stacked-column.component.html',\n  styleUrls: ['./stacked-column.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedColumnComponent {\n  private readonly input$ = new BehaviorSubject<Stack[]>([]);\n  @Input()\n  public set input(v: Stack[]) {\n    this.input$.next(v);\n  }\n  public get input(): Stack[] {\n    return this.input$.getValue();\n  }\n\n  private readonly width$ = new BehaviorSubject<number>(960);\n  @Input()\n  public set width(v: number) {\n    this.width$.next(v);\n  }\n  public get width(): number {\n    return this.width$.getValue();\n  }\n\n  private readonly height$ = new BehaviorSubject<number>(256);\n  @Input()\n  public set height(v: number) {\n    this.height$.next(v);\n  }\n  public get height(): number {\n    return this.height$.getValue();\n  }\n\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  readonly drawData$ = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      const viewBox = `0 0 ${width} ${height}`;\n      const padding = 8;\n\n      const [minDate, maxDate] = getDateRange(input);\n      const xSteps = timeMonth.range(\n        timeMonth.floor(minDate),\n        timeMonth.floor(timeMonth.offset(maxDate, 1)),\n      );\n      const xFormatter = timeFormat('%b');\n      const xOuterScale = scaleBand()\n        .domain(xSteps.map(date => xFormatter(date)))\n        .range([0, width]);\n\n      const columnWidth = 10;\n      const columnPadding = 4;\n      const outerPadding =\n        (xOuterScale.bandwidth() -\n          (columnPadding * (input.length - 1) + columnWidth * input.length)) /\n        2;\n\n      type SeriesPoint = Value & { series: Series };\n\n      const renderInformation = input.map(stackData => {\n        const stackedTable = stackData.reduce<SeriesPoint[]>(\n          (table, s) => [\n            ...table,\n            ...s.data.map(point => ({\n              ...point,\n              series: s,\n            })),\n          ],\n          [],\n        );\n        const indexedTable: Map<Date, Map<string, SeriesPoint>> = index(\n          stackedTable,\n          d => d.date,\n          d => d.series.label,\n        );\n\n        const stacked = stack<\n          typeof indexedTable extends Map<infer K, infer V> ? [K, V] : never\n        >()\n          .keys(stackData.map(v => v.label))\n          .value(([_, group], key) => {\n            return group.get(key)!.value;\n          })(indexedTable);\n\n        return { indexedTable, stacked };\n      });\n\n      let domainMax = 0;\n      renderInformation.forEach(({ stacked, indexedTable }) => {\n        [...indexedTable.values()].forEach((pointMap, i) => {\n          [...pointMap.values()].forEach((_, j) => {\n            domainMax = Math.max(domainMax, ...stacked[j][i]);\n          });\n        });\n      });\n      const yScale = scaleLinear()\n        .domain([0, domainMax])\n        .range([height - padding * 2, padding]);\n\n      const rects: {\n        dateValue: number;\n        x: number;\n        y: number;\n        width: number;\n        height: number;\n        fill: string;\n        striped: boolean;\n        seriesPoint: SeriesPoint;\n      }[] = [];\n      renderInformation.forEach(({ stacked, indexedTable }, index) => {\n        [...indexedTable.entries()].forEach(([date, pointMap], i) => {\n          [...pointMap.entries()].forEach(([_, seriesPoint], j) => {\n            const [baseline, topline] = stacked[j][i];\n\n            const width = columnWidth;\n            const height = yScale(baseline) - yScale(topline);\n            const x =\n              (xOuterScale(xFormatter(date)) ?? 0) +\n              outerPadding +\n              columnWidth * index +\n              columnPadding * index;\n            const y = yScale(baseline) - height;\n            const fill = `var(${seriesPoint.series.colorToken})`;\n\n            rects.push({\n              dateValue: date.valueOf(),\n              x,\n              y,\n              width,\n              height,\n              fill,\n              striped: seriesPoint.series.style === 'striped',\n              seriesPoint,\n            });\n          });\n        });\n      });\n\n      const xTicks = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        label: xFormatter(date),\n        x: (xOuterScale(xFormatter(date)) ?? 0) + xOuterScale.bandwidth() / 2,\n      }));\n      const yTicks = yScale.ticks(5);\n\n      const hoverBandWidth =\n        renderInformation.length * columnWidth +\n        (renderInformation.length + 1) * columnPadding;\n      const hoverBands = xSteps.map(date => ({\n        dateValue: date.valueOf(),\n        x:\n          (xOuterScale(xFormatter(date)) ?? 0) +\n          (xOuterScale.bandwidth() - hoverBandWidth) / 2,\n        y: yScale(domainMax),\n        width: hoverBandWidth,\n        height: yScale(0) - yScale(domainMax),\n      }));\n\n      return {\n        columnWidth,\n        hoverBands,\n        rects,\n        viewBox,\n        xTicks,\n        yScale,\n        yTicks,\n      };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly empty$ = this.input$.pipe(\n    map(input =>\n      input.every(stack => stack.every(series => series.data.length === 0)),\n    ),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  readonly hoveredBand$ = new BehaviorSubject<{\n    dateValue: number;\n    event: MouseEvent;\n  } | null>(null);\n\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand) return null;\n\n      const { target } = hoveredBand.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      type Metric = { label: string; value: string };\n      const metrics: Metric[] = drawData.rects\n        .filter(rect => rect.dateValue === hoveredBand.dateValue)\n        .map(rect => ({\n          label: rect.seriesPoint.series.label,\n          value: this.valueFormatter(rect.seriesPoint.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n","<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n  <svg\n    xmlns=\"http://www.w3.org/2000/svg\"\n    *ngIf=\"drawData$ | async; let d\"\n    [attr.viewBox]=\"d.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"d.columnWidth\"\n        [attr.height]=\"d.columnWidth\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          x1=\"0\"\n          [attr.y1]=\"d.columnWidth\"\n          [attr.x2]=\"d.columnWidth\"\n          y2=\"0\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let tick of d.xTicks\">\n      <rect\n        class=\"tick-background\"\n        [class.focused]=\"(hoveredBand$ | async)?.dateValue === tick.dateValue\"\n        [attr.x]=\"tick.x - 20\"\n        [attr.y]=\"d.yScale(0) + 3\"\n        width=\"40\"\n        height=\"12\"\n        rx=\"2\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"tick.x\"\n        [attr.y]=\"d.yScale(0) + 12\"\n        text-anchor=\"middle\"\n      >\n        {{ tick.label }}\n      </text>\n    </g>\n    <g *ngFor=\"let tick of d.yTicks\">\n      <rect\n        class=\"tick\"\n        x=\"0\"\n        [attr.y]=\"d.yScale(tick)\"\n        width=\"100%\"\n        height=\"1\"\n      ></rect>\n      <text class=\"tick-label\" x=\"0\" [attr.y]=\"d.yScale(tick)\" dy=\"-4\">\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g *ngFor=\"let rect of d.rects\">\n      <rect\n        class=\"column\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        [attr.fill]=\"rect.fill\"\n        [class.blurred]=\"\n          (hoveredBand$ | async) !== null &&\n          (hoveredBand$ | async)?.dateValue !== rect.dateValue\n        \"\n      ></rect>\n      <rect\n        *ngIf=\"rect.striped\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        fill=\"url(#stripes)\"\n      ></rect>\n    </g>\n    <rect\n      *ngFor=\"let rect of d.hoverBands\"\n      [attr.x]=\"rect.x\"\n      [attr.y]=\"rect.y\"\n      [attr.width]=\"rect.width\"\n      [attr.height]=\"rect.height\"\n      fill=\"transparent\"\n      (mouseenter)=\"\n        hoveredBand$.next({ dateValue: rect.dateValue, event: $event })\n      \"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    ></rect>\n  </svg>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n  <riv-callout\n    *riv-callout\n    [anchor]=\"callout.anchor\"\n    [isModal]=\"false\"\n    [preferredPosition]=\"'center-right'\"\n    [allowedPositions]=\"[\n      'center-right',\n      'center-left',\n      'top-center',\n      'bottom-center'\n    ]\"\n  >\n    <div class=\"callout-content\">\n      <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n        <div>{{ metric.label }}</div>\n        <div class=\"callout-metric-value\">{{ metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state></riv-zero-state>\n</ng-template>\n"]}
|
|
@@ -1014,7 +1014,7 @@ class StackedColumnComponent {
|
|
|
1014
1014
|
const xOuterScale = scaleBand()
|
|
1015
1015
|
.domain(xSteps.map(date => xFormatter(date)))
|
|
1016
1016
|
.range([0, width]);
|
|
1017
|
-
const columnWidth =
|
|
1017
|
+
const columnWidth = 10;
|
|
1018
1018
|
const columnPadding = 4;
|
|
1019
1019
|
const outerPadding = (xOuterScale.bandwidth() -
|
|
1020
1020
|
(columnPadding * (input.length - 1) + columnWidth * input.length)) /
|