UNPKG

@clr/angular

Version:

Angular components for Clarity

191 lines 27 kB
/* * Copyright (c) 2016-2025 Broadcom. All Rights Reserved. * The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. * This software is released under MIT license. * The full license information can be found in LICENSE in the root directory of this project. */ import { Directive, HostBinding, HostListener, Input, KeyValueDiffers, } from '@angular/core'; import { HostWrapper } from '../../utils/host-wrapping/host-wrapper'; import { CONTROL_SUFFIX } from './abstract-control'; import { IfControlStateService } from './if-control-state/if-control-state.service'; import { ContainerIdService } from './providers/container-id.service'; import { ControlClassService } from './providers/control-class.service'; import { ControlIdService } from './providers/control-id.service'; import { MarkControlService } from './providers/mark-control.service'; import { NgControlService } from './providers/ng-control.service'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; export var CHANGE_KEYS; (function (CHANGE_KEYS) { CHANGE_KEYS["FORM"] = "form"; CHANGE_KEYS["MODEL"] = "model"; })(CHANGE_KEYS || (CHANGE_KEYS = {})); export class WrappedFormControl { // I lost way too much time trying to make this work without injecting the ViewContainerRef and the Injector, // I'm giving up. So we have to inject these two manually for now. constructor(vcr, wrapperType, injector, _ngControl, renderer, el) { this.vcr = vcr; this.wrapperType = wrapperType; this._ngControl = _ngControl; this.renderer = renderer; this.el = el; this.index = 0; this.subscriptions = []; this.additionalDiffer = new Map(); if (injector) { this.ngControlService = injector.get(NgControlService, null); this.ifControlStateService = injector.get(IfControlStateService, null); this.controlClassService = injector.get(ControlClassService, null); this.markControlService = injector.get(MarkControlService, null); this.differs = injector.get(KeyValueDiffers, null); } if (this.controlClassService) { this.controlClassService.initControlClass(renderer, el.nativeElement); } if (this.markControlService) { this.subscriptions.push(this.markControlService.touchedChange.subscribe(() => { this.markAsTouched(); })); } if (this.ngControlService) { this.subscriptions.push(this.ngControlService.helpersChange.subscribe((state) => { this.setAriaDescribedBy(state); })); } } get id() { return this._id; } set id(value) { this._id = value; if (this.controlIdService) { this.controlIdService.id = value; } } get hasAdditionalControls() { return this.additionalDiffer.size > 0; } ngOnInit() { this._containerInjector = new HostWrapper(this.wrapperType, this.vcr, this.index); this.controlIdService = this._containerInjector.get(ControlIdService); /** * not all containers will provide `ContainerIdService` */ this.containerIdService = this._containerInjector.get(ContainerIdService, null); if (this._id) { this.controlIdService.id = this._id; } else { this._id = this.controlIdService.id; } if (this.ngControlService && this._ngControl) { if (!this.ngControlService.control) { this.ngControl = this._ngControl; this.ngControlService.setControl(this.ngControl); this.differ = this.differs.find(this._ngControl).create(); } else { this.ngControl = this.ngControlService.control; this.ngControlService.addAdditionalControl(this._ngControl); this.additionalDiffer.set(this._ngControl, this.differs.find(this._ngControl).create()); } } } ngDoCheck() { this.triggerDoCheck(this.differ, this.ngControl); if (this.hasAdditionalControls) { for (const [ngControl, differ] of this.additionalDiffer) { this.triggerDoCheck(differ, ngControl); } } } ngOnDestroy() { this.subscriptions.forEach(sub => sub?.unsubscribe()); } triggerValidation() { if (this.ifControlStateService) { this.ifControlStateService.triggerStatusChange(); } } // @TODO This method has a try/catch due to an unknown issue that came when building the clrToggle feature // We need to figure out why this fails for the ClrToggle scenario but works for Date picker... // To see the error, remove the try/catch here and run the ClrToggle suite to see issues getting the container // injector in time, and this ONLY HAPPENS in tests and not in dev/prod mode. getProviderFromContainer(token, notFoundValue) { try { return this._containerInjector.get(token, notFoundValue); } catch (e) { return notFoundValue; } } triggerDoCheck(differ, ngControl) { if (differ) { const changes = differ.diff(ngControl); if (changes) { changes.forEachChangedItem(change => { if ((change.key === CHANGE_KEYS.FORM || change.key === CHANGE_KEYS.MODEL) && change.currentValue !== change.previousValue) { this.triggerValidation(); } }); } } } markAsTouched() { if (this.ngControl) { this.ngControl.control.markAsTouched(); this.ngControl.control.updateValueAndValidity(); } if (this.ngControlService && this.ngControlService.hasAdditionalControls) { this.ngControlService.additionalControls?.forEach((ngControl) => { ngControl.control.markAsTouched(); ngControl.control.updateValueAndValidity(); }); } } setAriaDescribedBy(helpers) { if (helpers.show) { const ariaDescribedBy = this.getAriaDescribedById(helpers); if (ariaDescribedBy !== null) { this.renderer.setAttribute(this.el.nativeElement, 'aria-describedby', ariaDescribedBy); return; } } this.renderer.removeAttribute(this.el.nativeElement, 'aria-describedby'); } getAriaDescribedById(helpers) { const elementId = this.containerIdService?.id || this.controlIdService?.id; /** * If ContainerIdService or ControlIdService are missing don't try to guess * Don't set anything. */ if (!elementId) { return null; } /** * As the helper text is now always visible. If we have error/success then we should use both ids. */ const describedByIds = [`${elementId}-${CONTROL_SUFFIX.HELPER}`]; if (helpers.showInvalid) { describedByIds.push(`${elementId}-${CONTROL_SUFFIX.ERROR}`); } else if (helpers.showValid) { describedByIds.push(`${elementId}-${CONTROL_SUFFIX.SUCCESS}`); } return describedByIds.join(' '); } } WrappedFormControl.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: WrappedFormControl, deps: [{ token: i0.ViewContainerRef }, { token: i0.Type }, { token: i0.Injector }, { token: i1.NgControl }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive }); WrappedFormControl.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.2", type: WrappedFormControl, inputs: { id: "id" }, host: { listeners: { "blur": "triggerValidation()" }, properties: { "id": "this.id" } }, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: WrappedFormControl, decorators: [{ type: Directive }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.Type }, { type: i0.Injector }, { type: i1.NgControl }, { type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { id: [{ type: Input }, { type: HostBinding }], triggerValidation: [{ type: HostListener, args: ['blur'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wrapped-control.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/common/wrapped-control.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport {\n  Directive,\n  DoCheck,\n  ElementRef,\n  HostBinding,\n  HostListener,\n  InjectionToken,\n  Injector,\n  Input,\n  KeyValueDiffer,\n  KeyValueDiffers,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n  Type,\n  ViewContainerRef,\n} from '@angular/core';\nimport { NgControl } from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { HostWrapper } from '../../utils/host-wrapping/host-wrapper';\nimport { CONTROL_SUFFIX } from './abstract-control';\nimport { IfControlStateService } from './if-control-state/if-control-state.service';\nimport { ContainerIdService } from './providers/container-id.service';\nimport { ControlClassService } from './providers/control-class.service';\nimport { ControlIdService } from './providers/control-id.service';\nimport { MarkControlService } from './providers/mark-control.service';\nimport { Helpers, NgControlService } from './providers/ng-control.service';\n\nexport enum CHANGE_KEYS {\n  FORM = 'form',\n  MODEL = 'model',\n}\n\n@Directive()\nexport class WrappedFormControl<W> implements OnInit, DoCheck, OnDestroy {\n  _id: string;\n\n  protected controlIdService: ControlIdService;\n  protected ngControlService: NgControlService;\n  protected index = 0;\n  protected subscriptions: Subscription[] = [];\n\n  private ifControlStateService: IfControlStateService;\n  private controlClassService: ControlClassService;\n  private markControlService: MarkControlService;\n  private containerIdService: ContainerIdService;\n  private _containerInjector: Injector;\n  private differs: KeyValueDiffers;\n  private differ: KeyValueDiffer<any, any>;\n  private additionalDiffer = new Map<NgControl, KeyValueDiffer<any, any>>();\n  private ngControl: NgControl | null;\n\n  // I lost way too much time trying to make this work without injecting the ViewContainerRef and the Injector,\n  // I'm giving up. So we have to inject these two manually for now.\n  constructor(\n    protected vcr: ViewContainerRef,\n    protected wrapperType: Type<W>,\n    injector: Injector,\n    private _ngControl: NgControl | null,\n    protected renderer: Renderer2,\n    protected el: ElementRef<HTMLElement>\n  ) {\n    if (injector) {\n      this.ngControlService = injector.get(NgControlService, null);\n      this.ifControlStateService = injector.get(IfControlStateService, null);\n      this.controlClassService = injector.get(ControlClassService, null);\n      this.markControlService = injector.get(MarkControlService, null);\n      this.differs = injector.get(KeyValueDiffers, null);\n    }\n\n    if (this.controlClassService) {\n      this.controlClassService.initControlClass(renderer, el.nativeElement);\n    }\n    if (this.markControlService) {\n      this.subscriptions.push(\n        this.markControlService.touchedChange.subscribe(() => {\n          this.markAsTouched();\n        })\n      );\n    }\n\n    if (this.ngControlService) {\n      this.subscriptions.push(\n        this.ngControlService.helpersChange.subscribe((state: Helpers) => {\n          this.setAriaDescribedBy(state);\n        })\n      );\n    }\n  }\n\n  @Input()\n  @HostBinding()\n  get id() {\n    return this._id;\n  }\n  set id(value: string) {\n    this._id = value;\n    if (this.controlIdService) {\n      this.controlIdService.id = value;\n    }\n  }\n\n  private get hasAdditionalControls() {\n    return this.additionalDiffer.size > 0;\n  }\n\n  ngOnInit() {\n    this._containerInjector = new HostWrapper(this.wrapperType, this.vcr, this.index);\n    this.controlIdService = this._containerInjector.get(ControlIdService);\n\n    /**\n     * not all containers will provide `ContainerIdService`\n     */\n    this.containerIdService = this._containerInjector.get(ContainerIdService, null);\n\n    if (this._id) {\n      this.controlIdService.id = this._id;\n    } else {\n      this._id = this.controlIdService.id;\n    }\n\n    if (this.ngControlService && this._ngControl) {\n      if (!this.ngControlService.control) {\n        this.ngControl = this._ngControl;\n        this.ngControlService.setControl(this.ngControl);\n        this.differ = this.differs.find(this._ngControl).create();\n      } else {\n        this.ngControl = this.ngControlService.control;\n        this.ngControlService.addAdditionalControl(this._ngControl);\n        this.additionalDiffer.set(this._ngControl, this.differs.find(this._ngControl).create());\n      }\n    }\n  }\n\n  ngDoCheck() {\n    this.triggerDoCheck(this.differ, this.ngControl);\n    if (this.hasAdditionalControls) {\n      for (const [ngControl, differ] of this.additionalDiffer) {\n        this.triggerDoCheck(differ, ngControl);\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this.subscriptions.forEach(sub => sub?.unsubscribe());\n  }\n\n  @HostListener('blur')\n  triggerValidation() {\n    if (this.ifControlStateService) {\n      this.ifControlStateService.triggerStatusChange();\n    }\n  }\n\n  // @TODO This method has a try/catch due to an unknown issue that came when building the clrToggle feature\n  // We need to figure out why this fails for the ClrToggle scenario but works for Date picker...\n  // To see the error, remove the try/catch here and run the ClrToggle suite to see issues getting the container\n  // injector in time, and this ONLY HAPPENS in tests and not in dev/prod mode.\n  protected getProviderFromContainer<T>(token: Type<T> | InjectionToken<T>, notFoundValue?: T): T {\n    try {\n      return this._containerInjector.get(token, notFoundValue);\n    } catch (e) {\n      return notFoundValue;\n    }\n  }\n\n  private triggerDoCheck(differ, ngControl) {\n    if (differ) {\n      const changes = differ.diff(ngControl);\n      if (changes) {\n        changes.forEachChangedItem(change => {\n          if (\n            (change.key === CHANGE_KEYS.FORM || change.key === CHANGE_KEYS.MODEL) &&\n            change.currentValue !== change.previousValue\n          ) {\n            this.triggerValidation();\n          }\n        });\n      }\n    }\n  }\n\n  private markAsTouched(): void {\n    if (this.ngControl) {\n      this.ngControl.control.markAsTouched();\n      this.ngControl.control.updateValueAndValidity();\n    }\n    if (this.ngControlService && this.ngControlService.hasAdditionalControls) {\n      this.ngControlService.additionalControls?.forEach((ngControl: NgControl) => {\n        ngControl.control.markAsTouched();\n        ngControl.control.updateValueAndValidity();\n      });\n    }\n  }\n\n  private setAriaDescribedBy(helpers: Helpers) {\n    if (helpers.show) {\n      const ariaDescribedBy = this.getAriaDescribedById(helpers);\n      if (ariaDescribedBy !== null) {\n        this.renderer.setAttribute(this.el.nativeElement, 'aria-describedby', ariaDescribedBy);\n        return;\n      }\n    }\n\n    this.renderer.removeAttribute(this.el.nativeElement, 'aria-describedby');\n  }\n\n  private getAriaDescribedById(helpers: Helpers): string | null {\n    const elementId = this.containerIdService?.id || this.controlIdService?.id;\n    /**\n     * If ContainerIdService or ControlIdService are missing don't try to guess\n     * Don't set anything.\n     */\n    if (!elementId) {\n      return null;\n    }\n\n    /**\n     * As the helper text is now always visible. If we have error/success then we should use both ids.\n     */\n    const describedByIds = [`${elementId}-${CONTROL_SUFFIX.HELPER}`];\n    if (helpers.showInvalid) {\n      describedByIds.push(`${elementId}-${CONTROL_SUFFIX.ERROR}`);\n    } else if (helpers.showValid) {\n      describedByIds.push(`${elementId}-${CONTROL_SUFFIX.SUCCESS}`);\n    }\n    return describedByIds.join(' ');\n  }\n}\n"]}