@clr/angular
Version:
Angular components for Clarity
135 lines • 18.8 kB
JavaScript
/*
* 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"]}