@rivet-health/design-system 5.2.0 → 5.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -43,11 +43,12 @@ export class StackedRowComponent {
43
43
  this.SECTION_HEIGHTS.X_AXIS_LABEL -
44
44
  this.SECTION_PADDING.OUTER_TOP_BOTTOM -
45
45
  this.SECTION_PADDING.RECTANGLE_TO_ROW) /
46
- this.input.length; // calculate base row height
46
+ this.input.length -
47
+ this.SECTION_PADDING.BETWEEN_ROW; // calculate base row height
47
48
  // Clamping the row height between the min and max heights
48
49
  return Math.min(Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
49
50
  }
50
- return (this.RECTANGLE_HEIGHTS.MIN_HEIGHT + this.SECTION_PADDING.BETWEEN_ROW); // Default if no input
51
+ return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input
51
52
  },
52
53
  // Max Height of each rectangle
53
54
  PER_RECTANGLE: () => {
@@ -95,7 +96,9 @@ export class StackedRowComponent {
95
96
  : this.SECTION_HEIGHTS.X_AXIS_LABEL +
96
97
  this.SECTION_PADDING.OUTER_TOP_BOTTOM +
97
98
  this.SECTION_PADDING.RECTANGLE_TO_ROW +
98
- this.SECTION_HEIGHTS.PER_ROW() * input.length);
99
+ (this.SECTION_HEIGHTS.PER_ROW() +
100
+ this.SECTION_PADDING.BETWEEN_ROW) *
101
+ input.length);
99
102
  }), shareReplay({ refCount: true, bufferSize: 1 }));
100
103
  // Lets us know whether to display the zero data state
101
104
  this.empty$ = this.input$.pipe(map(input => input.every(stack => stack.data.length === 0)), shareReplay({ refCount: true, bufferSize: 1 }));
@@ -276,4 +279,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
276
279
  }], zeroStateMessage: [{
277
280
  type: Input
278
281
  }] } });
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"]}
282
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stacked-row.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.ts","../../../../../../projects/riv/src/lib/visualization/stacked-row/stacked-row.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input } from '@angular/core';\nimport { InternSet } from 'd3-array';\nimport { ScaleLinear, scaleLinear } from 'd3-scale';\nimport {\n  BehaviorSubject,\n  Observable,\n  combineLatest,\n  map,\n  shareReplay,\n} from 'rxjs';\nimport {\n  RowDisplayInfo,\n  RowLabel,\n  RowRectangle,\n  StackedRowDrawData,\n  getKeyOrder,\n  getLegendItems,\n  getRowDisplayInfo,\n  standardizeRowStacks,\n} from './stacked-row.helpers';\n\ntype CalloutMetric = {\n  rowName: string;\n  elementLabel: string;\n  value: string;\n};\n\n// TODO: once we upgrade to Angular 16, this component can be cleaned up with\n// signals instead of RxJS.\n@Component({\n  selector: 'riv-stacked-row',\n  templateUrl: './stacked-row.component.html',\n  styleUrls: ['./stacked-row.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StackedRowComponent {\n  // This is the list of rows we will display, in the order we will display them.\n  private readonly input$ = new BehaviorSubject<StackedRowComponent.RowStack[]>(\n    [],\n  );\n  @Input()\n  public set input(v: StackedRowComponent.RowStack[]) {\n    this.input$.next(v);\n  }\n  public get input(): StackedRowComponent.RowStack[] {\n    return this.input$.getValue();\n  }\n\n  // Width helps determine the size of the viewbox this graph is displaying in.\n  private readonly width$ = new BehaviorSubject<number>(960);\n  @Input()\n  public set width(v: number) {\n    this.width$.next(v);\n  }\n  public get width(): number {\n    return this.width$.getValue();\n  }\n\n  // Height helps determine the size of the viewbox this graph is displaying in.\n  private readonly height$ = new BehaviorSubject<number | null>(null);\n  @Input()\n  public set height(v: number | null) {\n    this.height$.next(v);\n  }\n  public get height(): number | null {\n    return this.height$.getValue();\n  }\n\n  // ValueFormatter determines how we style the values in a metric and the x-axis\n  @Input()\n  public valueFormatter: (v: number) => string = v => v.toString();\n\n  private readonly zeroStateMessage$ = new BehaviorSubject<string>(\n    'No data to display.',\n  );\n  @Input()\n  public set zeroStateMessage(v: string) {\n    this.zeroStateMessage$.next(v);\n  }\n  public get zeroStateMessage(): string {\n    return this.zeroStateMessage$.getValue();\n  }\n\n  /*\n    The layout we are shooting for is:\n\n              |   X Axis Labels  |\n    Row Name  |   Rectangles     |   Totals\n  */\n  readonly RECTANGLE_HEIGHTS = {\n    MIN_HEIGHT: 12, // Sets the smallest size a row can shrink to\n    MAX_HEIGHT: 42, // Sets the largest size a row can expand to\n  };\n\n  readonly SECTION_HEIGHTS = {\n    // Space reserved for the x-axis labels\n    X_AXIS_LABEL: 14,\n    // Height of each row\n    PER_ROW: (): number => {\n      //Only use height if specified\n      if (this.input.length && this.height) {\n        let rowHeight =\n          (this.height -\n            this.SECTION_HEIGHTS.X_AXIS_LABEL -\n            this.SECTION_PADDING.OUTER_TOP_BOTTOM -\n            this.SECTION_PADDING.RECTANGLE_TO_ROW) /\n            this.input.length -\n          this.SECTION_PADDING.BETWEEN_ROW; // calculate base row height\n        // Clamping the row height between the min and max heights\n        return Math.min(\n          Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT),\n          this.RECTANGLE_HEIGHTS.MAX_HEIGHT,\n        );\n      }\n      return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input\n    },\n    // Max Height of each rectangle\n    PER_RECTANGLE: (): number => {\n      //Only use height if specified\n      if (this.input.length && this.height) {\n        let rectangleHeight =\n          (this.height -\n            this.SECTION_HEIGHTS.X_AXIS_LABEL -\n            this.SECTION_PADDING.OUTER_TOP_BOTTOM -\n            this.SECTION_PADDING.RECTANGLE_TO_ROW) /\n            this.input.length -\n          this.SECTION_PADDING.BETWEEN_ROW; // calculate base rectangle height\n        // Clamping the rectangle height between the min and max heights\n        return Math.min(\n          Math.max(rectangleHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT),\n          this.RECTANGLE_HEIGHTS.MAX_HEIGHT,\n        );\n      }\n      return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input\n    },\n  };\n\n  readonly SECTION_WIDTHS = {\n    // Space reserved for the name at the start of the row\n    ROW_NAME: 100,\n    // Space reserved for the totals at the end of the row\n    ROW_TOTAL_LABEL: 72,\n  };\n\n  readonly SECTION_PADDING = {\n    // Padding between each edge of the viewbox and the content\n    OUTER_TOP_BOTTOM: 4,\n    OUTER_LEFT_RIGHT: 4,\n    // Padding between the bottom of one row and the top of the next row\n    BETWEEN_ROW: 4,\n    // Space between the row and the top of a rectangle in that row\n    RECTANGLE_TO_ROW: 6,\n  };\n\n  // The number of X-axis segments to create, with labels at the start of each segment.\n  readonly X_TICK_COUNT_MAX = 8;\n  readonly X_TICK_COUNT_MIN = 3;\n  readonly X_TICK_MAX_WIDTH = 900;\n  readonly X_TICK_MIN_WIDTH = 400;\n\n  // Zero on the X axis is after the left padding and the row name.\n  locationOfXAxisZero() {\n    return this.SECTION_PADDING.OUTER_LEFT_RIGHT + this.SECTION_WIDTHS.ROW_NAME;\n  }\n  // The end of the X axis is right before the row total and the right padding.\n  locationOfXAxisEnd(width: number) {\n    const spaceOnRight: number =\n      this.SECTION_WIDTHS.ROW_TOTAL_LABEL +\n      this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n    return width - spaceOnRight;\n  }\n  // The y-index of the x-axis ticks\n  locationOfXAxisTick() {\n    return (\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL\n    );\n  }\n  // The y-index of the top of the current row\n  getTopOfRow(rowIndex: number) {\n    const spaceAboveRows: number =\n      this.SECTION_PADDING.OUTER_TOP_BOTTOM + this.SECTION_HEIGHTS.X_AXIS_LABEL;\n\n    const aboveRows: number = rowIndex * this.SECTION_HEIGHTS.PER_ROW();\n    const aboveRowPaddings: number =\n      rowIndex * this.SECTION_PADDING.BETWEEN_ROW;\n    const allAboveRows: number = aboveRows + aboveRowPaddings;\n\n    return spaceAboveRows + allAboveRows;\n  }\n\n  // Calculate all the values we need for displaying the graph on screen.\n  getDrawnData(\n    rowStacks: StackedRowComponent.RowStack[],\n    width: number,\n    height: number,\n  ) {\n    if (rowStacks.length == 0) {\n      return null;\n    }\n    const viewBox = `0 0 ${width} ${height}`;\n\n    const keysInOrder: InternSet<string> = getKeyOrder(rowStacks);\n    const standardizedRows: StackedRowComponent.RowStack[] =\n      standardizeRowStacks(rowStacks, keysInOrder);\n    const rowsForDisplay: RowDisplayInfo[] = standardizedRows.map(stackData => {\n      return getRowDisplayInfo(stackData);\n    });\n\n    // The maximum value of the domain should be the highest total value from the rows.\n    const domainMax = Math.max(0, ...rowsForDisplay.map(row => row.rowTotal));\n    const xScale: ScaleLinear<number, number, never> = scaleLinear()\n      .domain([0, domainMax])\n      .range([this.locationOfXAxisZero(), this.locationOfXAxisEnd(width)]);\n\n    // Setup a linear scale for tick count based on width\n    const tickScale = scaleLinear()\n      .domain([this.X_TICK_MIN_WIDTH, this.X_TICK_MAX_WIDTH])\n      .range([this.X_TICK_COUNT_MIN, this.X_TICK_COUNT_MAX])\n      .clamp(true); // Ensures the scale does not go beyond the defined range\n\n    const xtickCountAdjusted = Math.round(tickScale(width));\n    const xTicks = xScale.ticks(xtickCountAdjusted);\n\n    const rects: RowRectangle[] = this.createRectanglesFromRows(\n      rowsForDisplay,\n      xScale,\n    );\n\n    const rowNamesToDraw: RowLabel[] = this.getRowNamesToDraw(rowsForDisplay);\n    const rowTotalsToDraw: RowLabel[] = this.getRowTotalsToDraw(\n      rowsForDisplay,\n      width,\n    );\n\n    const legendItems = getLegendItems(rowsForDisplay, keysInOrder);\n\n    return {\n      legendItems: legendItems,\n      rowHeight: this.SECTION_HEIGHTS.PER_RECTANGLE(),\n      rects: rects,\n      rowNames: rowNamesToDraw,\n      rowTotals: rowTotalsToDraw,\n      viewBox: viewBox,\n      xScale: xScale,\n      xTicks: xTicks,\n    };\n  }\n\n  // Get data for all the rectangles for all the rows\n  createRectanglesFromRows(\n    displayRows: RowDisplayInfo[],\n    xScale: ScaleLinear<number, number, never>,\n  ): RowRectangle[] {\n    const rectangles: RowRectangle[] = [];\n    displayRows.forEach((currentRow, rowIndex) => {\n      currentRow.elementsButStacked.forEach((currentElement, _) => {\n        const topOfRow: number = this.getTopOfRow(rowIndex);\n\n        const rectangleWidth: number =\n          xScale(currentElement.topline) - xScale(currentElement.baseline);\n        const rectangleHeight: number = this.SECTION_HEIGHTS.PER_RECTANGLE();\n\n        const rectangleX: number = xScale(currentElement.baseline);\n        const rectangleY: number =\n          topOfRow + this.SECTION_PADDING.RECTANGLE_TO_ROW;\n\n        const fill = `var(${currentElement.colorToken})`;\n\n        rectangles.push({\n          x: rectangleX,\n          y: rectangleY,\n          width: rectangleWidth,\n          height: rectangleHeight,\n          fill: fill,\n          striped: currentElement.style === 'striped',\n          rowName: currentRow.rowName,\n          stackedRowElement: currentElement,\n        });\n      });\n    });\n    return rectangles;\n  }\n\n  // Get data on the row names we are drawing\n  getRowNamesToDraw(displayRows: RowDisplayInfo[]): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row name looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;\n\n      return {\n        label: row.rowName,\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  // Get data on the row totals we are drawing\n  getRowTotalsToDraw(displayRows: RowDisplayInfo[], width: number): RowLabel[] {\n    return displayRows.map((row, rowIndex) => {\n      const rowLabelX = width - this.SECTION_PADDING.OUTER_LEFT_RIGHT;\n      // The row totals looks right if its base is at the base of the associated rectangle\n      const rowLabelY =\n        this.getTopOfRow(rowIndex) +\n        this.SECTION_PADDING.RECTANGLE_TO_ROW +\n        this.SECTION_HEIGHTS.PER_RECTANGLE() / 2;\n\n      return {\n        label: this.valueFormatter(row.rowTotal),\n        x: rowLabelX,\n        y: rowLabelY,\n      };\n    });\n  }\n\n  readonly drawData$: Observable<StackedRowDrawData | null> = combineLatest([\n    this.input$,\n    this.width$,\n    this.height$,\n  ]).pipe(\n    map(([input, width, height]) => {\n      return this.getDrawnData(\n        input,\n        width,\n        height\n          ? height\n          : this.SECTION_HEIGHTS.X_AXIS_LABEL +\n              this.SECTION_PADDING.OUTER_TOP_BOTTOM +\n              this.SECTION_PADDING.RECTANGLE_TO_ROW +\n              (this.SECTION_HEIGHTS.PER_ROW() +\n                this.SECTION_PADDING.BETWEEN_ROW) *\n                input.length,\n      );\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // Lets us know whether to display the zero data state\n  readonly empty$ = this.input$.pipe(\n    map(input => input.every(stack => stack.data.length === 0)),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n\n  // What rectangle we are currently hovering over\n  readonly hoveredRectangle$ = new BehaviorSubject<{\n    rectangle: RowRectangle;\n    event: MouseEvent;\n  } | null>(null);\n\n  // What band we are currently hovering over\n  readonly hoveredBand$ = new BehaviorSubject<{\n    rowName: string;\n    event: MouseEvent;\n  } | null>(null);\n\n  // Create the metric to display based on the currently hovered rectangle\n  readonly callout$ = combineLatest([this.hoveredBand$, this.drawData$]).pipe(\n    map(([hoveredBand, drawData]) => {\n      if (!hoveredBand || !drawData) return null;\n\n      const { target } = hoveredBand.event;\n      if (!target) return null;\n      const anchor = (<SVGRectElement>target).getBoundingClientRect();\n\n      type Metric = { label: string; value: string };\n      const metrics: Metric[] = drawData.rects\n        .filter(rect => rect.rowName === hoveredBand.rowName)\n        .map(rect => ({\n          label: rect.stackedRowElement.label,\n          value: this.valueFormatter(rect.stackedRowElement.value),\n        }));\n\n      return { anchor, metrics };\n    }),\n    shareReplay({ refCount: true, bufferSize: 1 }),\n  );\n}\n\nexport namespace StackedRowComponent {\n  // A specifically colored section of one row.\n  export type RowElement = {\n    label: string; // Legend element name\n    value: number;\n    colorToken: string;\n    style?: 'solid' | 'striped';\n  };\n  // The input that determines one row.\n  export type RowStack = {\n    rowName: string;\n    data: RowElement[];\n  };\n}\n","<div *ngIf=\"!(empty$ | async); else zeroState\" class=\"container\">\n  <svg\n    xmlns=\"http://www.w3.org/2000/svg\"\n    *ngIf=\"drawData$ | async; let draw\"\n    [attr.viewBox]=\"draw.viewBox\"\n  >\n    <defs>\n      <pattern\n        id=\"stripes\"\n        x=\"0\"\n        y=\"0\"\n        [attr.width]=\"draw.rowHeight\"\n        [attr.height]=\"draw.rowHeight\"\n        patternUnits=\"userSpaceOnUse\"\n      >\n        <line\n          [attr.x1]=\"0\"\n          [attr.y1]=\"0\"\n          [attr.x2]=\"draw.rowHeight\"\n          [attr.y2]=\"draw.rowHeight\"\n          stroke=\"var(--white-100)\"\n          stroke-width=\"1\"\n        ></line>\n      </pattern>\n    </defs>\n    <g *ngFor=\"let rowNameToDraw of draw.rowNames\">\n      <text\n        rivSVGTextTruncate\n        [text]=\"rowNameToDraw.label\"\n        [width]=\"SECTION_WIDTHS.ROW_NAME\"\n        class=\"row-title\"\n        [attr.x]=\"rowNameToDraw.x\"\n        [attr.y]=\"rowNameToDraw.y\"\n      ></text>\n    </g>\n    <g *ngFor=\"let tick of draw.xTicks\">\n      <rect\n        class=\"tick\"\n        [attr.x]=\"draw.xScale(tick)\"\n        y=\"0\"\n        width=\"1\"\n        height=\"100%\"\n      ></rect>\n      <text\n        class=\"tick-label\"\n        [attr.x]=\"draw.xScale(tick)\"\n        [attr.y]=\"locationOfXAxisTick()\"\n        dx=\"4\"\n        dy=\"-4\"\n      >\n        {{ valueFormatter(tick) }}\n      </text>\n    </g>\n    <g\n      *ngFor=\"let rect of draw.rects\"\n      (mouseenter)=\"hoveredBand$.next({ rowName: rect.rowName, event: $event })\"\n      (mouseleave)=\"hoveredBand$.next(null)\"\n    >\n      <rect\n        class=\"row\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        [attr.fill]=\"rect.fill\"\n        [class.blurred]=\"\n          (hoveredBand$ | async) !== null &&\n          (hoveredBand$ | async)?.rowName !== rect.rowName\n        \"\n      ></rect>\n      <rect\n        *ngIf=\"rect.striped\"\n        [attr.x]=\"rect.x\"\n        [attr.y]=\"rect.y\"\n        [attr.width]=\"rect.width\"\n        [attr.height]=\"rect.height\"\n        fill=\"url(#stripes)\"\n      ></rect>\n    </g>\n    <g *ngFor=\"let rowTotalToDraw of draw.rowTotals\">\n      <text\n        class=\"row-total\"\n        [attr.x]=\"rowTotalToDraw.x\"\n        [attr.y]=\"rowTotalToDraw.y\"\n        text-anchor=\"end\"\n      >\n        {{ rowTotalToDraw.label }}\n      </text>\n    </g>\n  </svg>\n  <legend *ngIf=\"drawData$ | async; let draw\">\n    <riv-legend-item\n      *ngFor=\"let item of draw.legendItems\"\n      [label]=\"item.label\"\n      [colorToken]=\"item.colorToken\"\n      [style]=\"item.style ? item.style : 'solid'\"\n    ></riv-legend-item>\n  </legend>\n</div>\n\n<ng-container *ngIf=\"callout$ | async; let callout\">\n  <riv-callout\n    *riv-callout\n    [anchor]=\"callout.anchor\"\n    [isModal]=\"false\"\n    [preferredPosition]=\"'bottom-center'\"\n    [allowedPositions]=\"[\n      'center-right',\n      'center-left',\n      'top-center',\n      'bottom-center'\n    ]\"\n  >\n    <div class=\"callout-content\">\n      <div class=\"callout-metric\" *ngFor=\"let metric of callout.metrics\">\n        <div>{{ metric.label }}</div>\n        <div class=\"callout-metric-value\">{{ metric.value }}</div>\n      </div>\n    </div>\n  </riv-callout>\n</ng-container>\n\n<ng-template #zeroState>\n  <riv-zero-state [message]=\"zeroStateMessage\"></riv-zero-state>\n</ng-template>\n"]}
@@ -3035,11 +3035,12 @@ class StackedRowComponent {
3035
3035
  this.SECTION_HEIGHTS.X_AXIS_LABEL -
3036
3036
  this.SECTION_PADDING.OUTER_TOP_BOTTOM -
3037
3037
  this.SECTION_PADDING.RECTANGLE_TO_ROW) /
3038
- this.input.length; // calculate base row height
3038
+ this.input.length -
3039
+ this.SECTION_PADDING.BETWEEN_ROW; // calculate base row height
3039
3040
  // Clamping the row height between the min and max heights
3040
3041
  return Math.min(Math.max(rowHeight, this.RECTANGLE_HEIGHTS.MIN_HEIGHT), this.RECTANGLE_HEIGHTS.MAX_HEIGHT);
3041
3042
  }
3042
- return (this.RECTANGLE_HEIGHTS.MIN_HEIGHT + this.SECTION_PADDING.BETWEEN_ROW); // Default if no input
3043
+ return this.RECTANGLE_HEIGHTS.MIN_HEIGHT; // Default if no input
3043
3044
  },
3044
3045
  // Max Height of each rectangle
3045
3046
  PER_RECTANGLE: () => {
@@ -3087,7 +3088,9 @@ class StackedRowComponent {
3087
3088
  : this.SECTION_HEIGHTS.X_AXIS_LABEL +
3088
3089
  this.SECTION_PADDING.OUTER_TOP_BOTTOM +
3089
3090
  this.SECTION_PADDING.RECTANGLE_TO_ROW +
3090
- this.SECTION_HEIGHTS.PER_ROW() * input.length);
3091
+ (this.SECTION_HEIGHTS.PER_ROW() +
3092
+ this.SECTION_PADDING.BETWEEN_ROW) *
3093
+ input.length);
3091
3094
  }), shareReplay({ refCount: true, bufferSize: 1 }));
3092
3095
  // Lets us know whether to display the zero data state
3093
3096
  this.empty$ = this.input$.pipe(map(input => input.every(stack => stack.data.length === 0)), shareReplay({ refCount: true, bufferSize: 1 }));