UNPKG

@clr/angular

Version:

Angular components for Clarity

135 lines 18.8 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 { ContentChild, Directive, Optional } from '@angular/core'; import { ClrControlError } from './error'; import { ClrControlHelper } from './helper'; import { CONTROL_STATE } from './if-control-state/if-control-state.service'; import { ClrLabel } from './label'; import { ClrControlSuccess } from './success'; import * as i0 from "@angular/core"; import * as i1 from "./if-control-state/if-control-state.service"; import * as i2 from "./providers/layout.service"; import * as i3 from "./providers/control-class.service"; import * as i4 from "./providers/ng-control.service"; export class ClrAbstractContainer { constructor(ifControlStateService, layoutService, controlClassService, ngControlService) { this.ifControlStateService = ifControlStateService; this.layoutService = layoutService; this.controlClassService = controlClassService; this.ngControlService = ngControlService; this.subscriptions = []; this.subscriptions.push(ifControlStateService.statusChanges.subscribe((state) => { this.state = state; // Make sure everything is updated before dispatching the values for helpers setTimeout(() => { this.updateHelpers(); }); })); this.subscriptions.push(ngControlService.controlChanges.subscribe(control => { this.control = control; }), ngControlService.additionalControlsChanges.subscribe(controls => { this.additionalControls = controls; })); } /** * @NOTE * Helper control is a bit different than the others, it must be always visible: * - Labels and instructions must always accompany forms and are persistent. * - The recommendation here is to always have helper text or anything instructions visible. * - The expectation is to have error text + helper text in the errored state. this way all users will have the helper text information always available. */ get showHelper() { /** * @NOTE * Saving the previous version in case something is changed. We'll return always true so we can be flexible * and keep the condition per components. * * return ( * Helper Component exist and the state of the form is NONE (not touched) * (!!this.controlHelperComponent && (!this.touched || this.state === CONTROL_STATE.NONE)) || * or there is no success component but the state of the form is VALID - show helper information * (!!this.controlSuccessComponent === false && this.state === CONTROL_STATE.VALID) || * or there is no error component but the state of the form is INVALID - show helper information * (!!this.controlErrorComponent === false && this.state === CONTROL_STATE.INVALID) * ); */ return Boolean(this.controlHelperComponent); } get showValid() { return this.touched && this.state === CONTROL_STATE.VALID && this.successMessagePresent; } get showInvalid() { return this.touched && this.state === CONTROL_STATE.INVALID && this.errorMessagePresent; } get successMessagePresent() { return !!this.controlSuccessComponent; } get errorMessagePresent() { return !!this.controlErrorComponent; } get touched() { return !!(this.control?.touched || this.additionalControls?.some(control => control.touched)); } ngAfterContentInit() { /** * We gonna set the helper control state, after all or most of the components * are ready - also this will trigger some initial flows into wrappers and controls, * like locating IDs and setting attributes. */ this.updateHelpers(); } ngOnDestroy() { this.subscriptions.forEach(subscription => subscription.unsubscribe()); } controlClass() { /** * Decide what subtext to display: * - container is valid but no success component is implemented - use helper class * - container is valid and success component is implemented - use success class */ if ((!this.controlSuccessComponent && this.state === CONTROL_STATE.VALID) || !this.touched) { return this.controlClassService.controlClass(CONTROL_STATE.NONE, this.addGrid()); } /** * Pass form control state and return string of classes to be applied to the container. */ return this.controlClassService.controlClass(this.state, this.addGrid()); } addGrid() { return this.layoutService && !this.layoutService.isVertical(); } updateHelpers() { if (this.ngControlService) { this.ngControlService.setHelpers({ show: this.showInvalid || this.showHelper || this.showValid, showInvalid: this.showInvalid, showHelper: this.showHelper, showValid: this.showValid, }); } } } ClrAbstractContainer.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrAbstractContainer, deps: [{ token: i1.IfControlStateService }, { token: i2.LayoutService, optional: true }, { token: i3.ControlClassService }, { token: i4.NgControlService }], target: i0.ɵɵFactoryTarget.Directive }); ClrAbstractContainer.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.2", type: ClrAbstractContainer, queries: [{ propertyName: "label", first: true, predicate: ClrLabel, descendants: true }, { propertyName: "controlSuccessComponent", first: true, predicate: ClrControlSuccess, descendants: true }, { propertyName: "controlErrorComponent", first: true, predicate: ClrControlError, descendants: true }, { propertyName: "controlHelperComponent", first: true, predicate: ClrControlHelper, descendants: true }], ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrAbstractContainer, decorators: [{ type: Directive }], ctorParameters: function () { return [{ type: i1.IfControlStateService }, { type: i2.LayoutService, decorators: [{ type: Optional }] }, { type: i3.ControlClassService }, { type: i4.NgControlService }]; }, propDecorators: { label: [{ type: ContentChild, args: [ClrLabel, { static: false }] }], controlSuccessComponent: [{ type: ContentChild, args: [ClrControlSuccess] }], controlErrorComponent: [{ type: ContentChild, args: [ClrControlError] }], controlHelperComponent: [{ type: ContentChild, args: [ClrControlHelper] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-container.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/common/abstract-container.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 { AfterContentInit, ContentChild, Directive, OnDestroy, Optional } from '@angular/core';\nimport { NgControl } from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { ClrControlError } from './error';\nimport { ClrControlHelper } from './helper';\nimport { CONTROL_STATE, IfControlStateService } from './if-control-state/if-control-state.service';\nimport { ClrLabel } from './label';\nimport { ControlClassService } from './providers/control-class.service';\nimport { LayoutService } from './providers/layout.service';\nimport { NgControlService } from './providers/ng-control.service';\nimport { ClrControlSuccess } from './success';\n\n@Directive()\nexport abstract class ClrAbstractContainer implements OnDestroy, AfterContentInit {\n  @ContentChild(ClrLabel, { static: false }) label: ClrLabel;\n  @ContentChild(ClrControlSuccess) controlSuccessComponent: ClrControlSuccess;\n  @ContentChild(ClrControlError) controlErrorComponent: ClrControlError;\n  @ContentChild(ClrControlHelper) controlHelperComponent: ClrControlHelper;\n\n  control: NgControl;\n  additionalControls: NgControl[];\n\n  protected subscriptions: Subscription[] = [];\n\n  private state: CONTROL_STATE;\n\n  constructor(\n    protected ifControlStateService: IfControlStateService,\n    @Optional() protected layoutService: LayoutService,\n    protected controlClassService: ControlClassService,\n    protected ngControlService: NgControlService\n  ) {\n    this.subscriptions.push(\n      ifControlStateService.statusChanges.subscribe((state: CONTROL_STATE) => {\n        this.state = state;\n        // Make sure everything is updated before dispatching the values for helpers\n        setTimeout(() => {\n          this.updateHelpers();\n        });\n      })\n    );\n\n    this.subscriptions.push(\n      ngControlService.controlChanges.subscribe(control => {\n        this.control = control;\n      }),\n      ngControlService.additionalControlsChanges.subscribe(controls => {\n        this.additionalControls = controls;\n      })\n    );\n  }\n\n  /**\n   * @NOTE\n   * Helper control is a bit different than the others, it must be always visible:\n   *   -  Labels and instructions must always accompany forms and are persistent.\n   *   -  The recommendation here is to always have helper text or anything instructions visible.\n   *   -  The expectation is to have error text + helper text in the errored state. this way all users will have the helper text information always available.\n   */\n  get showHelper(): boolean {\n    /**\n     * @NOTE\n     * Saving the previous version in case something is changed. We'll return always true so we can be flexible\n     * and keep the condition per components.\n     *\n     * return (\n     * Helper Component exist and the state of the form is NONE (not touched)\n     * (!!this.controlHelperComponent && (!this.touched || this.state === CONTROL_STATE.NONE)) ||\n     * or there is no success component but the state of the form is VALID - show helper information\n     * (!!this.controlSuccessComponent === false && this.state === CONTROL_STATE.VALID) ||\n     * or there is no error component but the state of the form is INVALID - show helper information\n     * (!!this.controlErrorComponent === false && this.state === CONTROL_STATE.INVALID)\n     * );\n     */\n    return Boolean(this.controlHelperComponent);\n  }\n\n  get showValid(): boolean {\n    return this.touched && this.state === CONTROL_STATE.VALID && this.successMessagePresent;\n  }\n\n  get showInvalid(): boolean {\n    return this.touched && this.state === CONTROL_STATE.INVALID && this.errorMessagePresent;\n  }\n\n  protected get successMessagePresent() {\n    return !!this.controlSuccessComponent;\n  }\n\n  protected get errorMessagePresent() {\n    return !!this.controlErrorComponent;\n  }\n\n  private get touched() {\n    return !!(this.control?.touched || this.additionalControls?.some(control => control.touched));\n  }\n\n  ngAfterContentInit() {\n    /**\n     * We gonna set the helper control state, after all or most of the components\n     * are ready - also this will trigger some initial flows into wrappers and controls,\n     * like locating IDs  and setting  attributes.\n     */\n    this.updateHelpers();\n  }\n\n  ngOnDestroy() {\n    this.subscriptions.forEach(subscription => subscription.unsubscribe());\n  }\n\n  controlClass() {\n    /**\n     * Decide what subtext to display:\n     *   - container is valid but no success component is implemented - use helper class\n     *   - container is valid and success component is implemented - use success class\n     */\n    if ((!this.controlSuccessComponent && this.state === CONTROL_STATE.VALID) || !this.touched) {\n      return this.controlClassService.controlClass(CONTROL_STATE.NONE, this.addGrid());\n    }\n    /**\n     * Pass form control state and return string of classes to be applied to the container.\n     */\n    return this.controlClassService.controlClass(this.state, this.addGrid());\n  }\n\n  addGrid() {\n    return this.layoutService && !this.layoutService.isVertical();\n  }\n\n  private updateHelpers() {\n    if (this.ngControlService) {\n      this.ngControlService.setHelpers({\n        show: this.showInvalid || this.showHelper || this.showValid,\n        showInvalid: this.showInvalid,\n        showHelper: this.showHelper,\n        showValid: this.showValid,\n      });\n    }\n  }\n}\n"]}