UNPKG

@engie-group/fluid-design-system-angular

Version:

Fluid Design System Angular

169 lines 19.4 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Input, Output, ViewChild } from '@angular/core'; import { Subject, takeUntil } from 'rxjs'; import { Utils } from '../../utils/utils.util'; import { SegmentedControlButtonComponent } from '../segmented-control-button/segmented-control-button.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class SegmentedControlComponent { /** * Selected value */ set value(value) { this._value = value; this.setActive(value); } get value() { return this._value; } /** * Overrides disabled state on all buttons */ set isDisabled(value) { this._isDisabled = value; this.setButtonsIsDisabled(); } get isDisabled() { return this._isDisabled; } constructor(cdr) { this.cdr = cdr; this.segmentedControlClass = 'nj-segmented-control'; /** * @ignore */ this.unsubscribe = new Subject(); /** * @ignore */ this._onChange = (_) => { }; /** * @ignore */ this._onTouched = () => { }; /** * Segmented Control size */ this.size = 'md'; /** * Output that emits value change */ this.valueChange = new EventEmitter(); } ngAfterViewInit() { // A render is being done after view init so setTimeout allows us to delay any attribute modification // in the call stack, so it can be taken account on next render only setTimeout(() => { this.segmentedButtons = this.segmentedButtonsList?.toArray(); this.setButtonsIsDisabled(); this.initializeClickListener(); if (this.value) { this.setActive(this.value); } }); } setButtonsIsDisabled() { if (!Utils.isUndefinedOrNull(this.isDisabled) && this.segmentedButtons) { for (const segmentedButton of this.segmentedButtons) { segmentedButton.setIsDisabled(this.isDisabled); } } } ngOnDestroy() { this.unsubscribe.next(); this.unsubscribe.complete(); } /** * @ignore */ initializeClickListener() { if (Utils.isUndefinedOrNull(this.segmentedButtons)) { return; } this.segmentedButtons.forEach((button, value) => { button.buttonClick.pipe(takeUntil(this.unsubscribe)).subscribe((_) => { button.isSelected = true; this.writeValue(button.value); }); }); } setActive(buttonValue) { if (!this.segmentedButtons) { return; } for (const button of this.segmentedButtons) { if (button?.value === buttonValue) { button.setIsSelected(true); const selectedBoundingRect = button.getClientBoundingRect(); const segmentedControlBoundingRect = this.segmentedControl.nativeElement.getBoundingClientRect(); const segmentedControlStyle = this.segmentedControl?.nativeElement?.style; segmentedControlStyle.setProperty('--nj-sc-btn-selected-width', `${selectedBoundingRect.width}px`); segmentedControlStyle.setProperty('--nj-sc-btn-selected-right', `${segmentedControlBoundingRect.right - selectedBoundingRect.right}px`); } else { button.setIsSelected(false); } } } /** * @ignore */ getSizeClass() { if (!this.size || this.size === 'md') { return; } return `${this.segmentedControlClass}--${this.size}`; } /** * @ignore */ writeValue(value) { if (value === this.value) { return; } this.setActive(value); this.value = value; } /** * @ignore */ registerOnChange(fn) { this._onChange = fn; } /** * @ignore */ registerOnTouched(fn) { this._onTouched = fn; } /** * @ignore */ setDisabledState(isDisabled) { this.isDisabled = isDisabled; this.cdr.markForCheck(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SegmentedControlComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: SegmentedControlComponent, isStandalone: true, selector: "nj-segmented-control", inputs: { value: "value", ariaLabel: "ariaLabel", isDisabled: "isDisabled", size: "size" }, outputs: { valueChange: "valueChange" }, queries: [{ propertyName: "segmentedButtonsList", predicate: SegmentedControlButtonComponent }], viewQueries: [{ propertyName: "segmentedControl", first: true, predicate: ["segmentedControl"], descendants: true }], ngImport: i0, template: "<div class=\"nj-segmented-control\" role=\"group\" [attr.aria-label]=\"ariaLabel\" [ngClass]=\"getSizeClass()\"\n #segmentedControl>\n <ng-content></ng-content>\n</div>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SegmentedControlComponent, decorators: [{ type: Component, args: [{ selector: 'nj-segmented-control', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [SegmentedControlButtonComponent, CommonModule], template: "<div class=\"nj-segmented-control\" role=\"group\" [attr.aria-label]=\"ariaLabel\" [ngClass]=\"getSizeClass()\"\n #segmentedControl>\n <ng-content></ng-content>\n</div>\n" }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { value: [{ type: Input }], ariaLabel: [{ type: Input }], isDisabled: [{ type: Input }], size: [{ type: Input }], segmentedButtonsList: [{ type: ContentChildren, args: [SegmentedControlButtonComponent] }], segmentedControl: [{ type: ViewChild, args: ['segmentedControl'] }], valueChange: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"segmented-control.component.js","sourceRoot":"","sources":["../../../../src/components/segmented-control/segmented-control.component.ts","../../../../src/components/segmented-control/segmented-control.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  QueryList,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor } from '@angular/forms';\nimport { Subject, takeUntil } from 'rxjs';\nimport { Utils } from '../../utils/utils.util';\nimport { SegmentedControlButtonComponent } from '../segmented-control-button/segmented-control-button.component';\nimport { SegmentedControlSize } from './segmented-control.model';\n\n@Component({\n  selector: 'nj-segmented-control',\n  templateUrl: './segmented-control.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [SegmentedControlButtonComponent, CommonModule]\n})\nexport class SegmentedControlComponent implements ControlValueAccessor, AfterViewInit, OnDestroy {\n  private readonly segmentedControlClass = 'nj-segmented-control';\n\n  /**\n   * @ignore\n   */\n  private unsubscribe: Subject<void> = new Subject<void>();\n\n  /**\n   * @ignore\n   */\n  private _onChange = (_: any): void => {};\n\n  /**\n   * @ignore\n   */\n  private _onTouched = (): void => {};\n\n  /**\n   * @ignore\n   */\n  private _isDisabled: boolean;\n\n  /**\n   * @ignore\n   */\n  private _value: string;\n\n  /**\n   * @ignore\n   */\n  public segmentedButtons: SegmentedControlButtonComponent[];\n\n  /**\n   * Selected value\n   */\n  @Input()\n  set value(value: string) {\n    this._value = value;\n    this.setActive(value);\n  }\n\n  get value(): string {\n    return this._value;\n  }\n\n  /**\n   * Selected aria label\n   */\n  @Input() ariaLabel: string;\n\n  /**\n   * Overrides disabled state on all buttons\n   */\n  @Input()\n  set isDisabled(value: boolean) {\n    this._isDisabled = value;\n    this.setButtonsIsDisabled();\n  }\n\n  get isDisabled(): boolean {\n    return this._isDisabled;\n  }\n\n  /**\n   * Segmented Control size\n   */\n  @Input() size: SegmentedControlSize = 'md';\n\n  /**\n   * @ignore\n   */\n  @ContentChildren(SegmentedControlButtonComponent) segmentedButtonsList: QueryList<SegmentedControlButtonComponent>;\n\n  /**\n   * @ignore\n   */\n  @ViewChild('segmentedControl') segmentedControl: ElementRef<HTMLDivElement>;\n\n  /**\n   * Output that emits value change\n   */\n  @Output() valueChange: EventEmitter<string> = new EventEmitter<string>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngAfterViewInit() {\n    // A render is being done after view init so setTimeout allows us to delay any attribute modification\n    // in the call stack, so it can be taken account on next render only\n    setTimeout(() => {\n      this.segmentedButtons = this.segmentedButtonsList?.toArray();\n      this.setButtonsIsDisabled();\n      this.initializeClickListener();\n      if (this.value) {\n        this.setActive(this.value);\n      }\n    });\n  }\n\n  private setButtonsIsDisabled() {\n    if (!Utils.isUndefinedOrNull(this.isDisabled) && this.segmentedButtons) {\n      for (const segmentedButton of this.segmentedButtons) {\n        segmentedButton.setIsDisabled(this.isDisabled);\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this.unsubscribe.next();\n    this.unsubscribe.complete();\n  }\n\n  /**\n   * @ignore\n   */\n  private initializeClickListener() {\n    if (Utils.isUndefinedOrNull(this.segmentedButtons)) {\n      return;\n    }\n    this.segmentedButtons.forEach((button, value) => {\n      button.buttonClick.pipe(takeUntil(this.unsubscribe)).subscribe((_) => {\n        button.isSelected = true;\n        this.writeValue(button.value);\n      });\n    });\n  }\n\n  private setActive(buttonValue: string) {\n    if (!this.segmentedButtons) {\n      return;\n    }\n    for (const button of this.segmentedButtons) {\n      if (button?.value === buttonValue) {\n        button.setIsSelected(true);\n        const selectedBoundingRect = button.getClientBoundingRect();\n        const segmentedControlBoundingRect = this.segmentedControl.nativeElement.getBoundingClientRect();\n        const segmentedControlStyle = this.segmentedControl?.nativeElement?.style;\n        segmentedControlStyle.setProperty('--nj-sc-btn-selected-width', `${selectedBoundingRect.width}px`);\n        segmentedControlStyle.setProperty(\n          '--nj-sc-btn-selected-right',\n          `${segmentedControlBoundingRect.right - selectedBoundingRect.right}px`\n        );\n      } else {\n        button.setIsSelected(false);\n      }\n    }\n  }\n\n  /**\n   * @ignore\n   */\n  getSizeClass(): string {\n    if (!this.size || this.size === 'md') {\n      return;\n    }\n    return `${this.segmentedControlClass}--${this.size}`;\n  }\n\n  /**\n   * @ignore\n   */\n  writeValue(value: any): void {\n    if (value === this.value) {\n      return;\n    }\n    this.setActive(value);\n    this.value = value;\n  }\n\n  /**\n   * @ignore\n   */\n  registerOnChange(fn: any): void {\n    this._onChange = fn;\n  }\n\n  /**\n   * @ignore\n   */\n  registerOnTouched(fn: any): void {\n    this._onTouched = fn;\n  }\n\n  /**\n   * @ignore\n   */\n  setDisabledState?(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n    this.cdr.markForCheck();\n  }\n}\n","<div class=\"nj-segmented-control\" role=\"group\" [attr.aria-label]=\"ariaLabel\" [ngClass]=\"getSizeClass()\"\n     #segmentedControl>\n  <ng-content></ng-content>\n</div>\n"]}