@rivet-health/design-system 5.0.2 → 5.1.2
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/fesm2015/rivet-health-design-system.mjs +3 -0
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +3 -0
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/lib/input/single-select/single-select.component.d.ts +1 -0
- 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":"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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  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"]}
|