@klippa/ngx-enhancy-forms 14.22.10 → 14.22.12

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.
@@ -35,6 +35,7 @@ export class FormElementComponent {
35
35
  this.errorMessages = DEFAULT_ERROR_MESSAGES;
36
36
  this.customErrorHandlers = [];
37
37
  this.popupState = 'onHover';
38
+ this.subscriptions = [];
38
39
  this.setErrorMessageIsTruncated = (isTruncated) => {
39
40
  this.errorFullyVisible = !isTruncated;
40
41
  };
@@ -51,10 +52,12 @@ export class FormElementComponent {
51
52
  async ngAfterViewInit() {
52
53
  await awaitableForNextCycle();
53
54
  this.fieldInput?.setTailTpl(this.tailTpl);
54
- this.fieldInput?.onTouch.asObservable().subscribe((e) => {
55
+ const subscription = this.fieldInput?.onTouch.asObservable().subscribe(() => {
55
56
  this.determinePopupState();
56
57
  });
57
- [...getAllLimitingContainers(this.elRef.nativeElement), window].forEach(e => e.addEventListener('scroll', this.setErrorTooltipOffset));
58
+ if (isValueSet(subscription)) {
59
+ this.subscriptions.push(subscription);
60
+ }
58
61
  }
59
62
  shouldShowErrorMessages() {
60
63
  return this.parent?.showErrorMessages !== false;
@@ -68,21 +71,40 @@ export class FormElementComponent {
68
71
  this.attachedControl = formControl;
69
72
  this.parent.registerControl(formControl, this);
70
73
  this.input = input;
71
- this.attachedControl.statusChanges.subscribe((e) => {
74
+ const subscription = this.attachedControl.statusChanges.subscribe(() => {
72
75
  this.determinePopupState();
73
76
  });
77
+ this.subscriptions.push(subscription);
74
78
  this.determinePopupState();
75
79
  }
76
80
  determinePopupState() {
81
+ const prevState = this.popupState;
77
82
  if (stringIsSetAndFilled(this.getErrorToShow())) {
78
83
  this.popupState = 'onHover';
79
- return;
80
84
  }
81
- if (isValueSet(this.getWarningToShow())) {
85
+ else if (isValueSet(this.getWarningToShow())) {
82
86
  this.popupState = 'lockedOpen';
87
+ }
88
+ else {
89
+ this.popupState = 'onHover';
90
+ }
91
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
92
+ }
93
+ setUpErrorTooltipListeners(prev, current) {
94
+ if (prev === current) {
83
95
  return;
84
96
  }
85
- this.popupState = 'onHover';
97
+ const containers = [...getAllLimitingContainers(this.elRef.nativeElement), window];
98
+ if (current === 'lockedOpen') {
99
+ containers.forEach(e => {
100
+ e.addEventListener('scroll', this.setErrorTooltipOffset);
101
+ });
102
+ }
103
+ else {
104
+ containers.forEach(e => {
105
+ e.removeEventListener('scroll', this.setErrorTooltipOffset);
106
+ });
107
+ }
86
108
  }
87
109
  unregisterControl(formControl) {
88
110
  this.attachedControl = null;
@@ -197,7 +219,9 @@ export class FormElementComponent {
197
219
  return isValueSet(this.getWarningToShow());
198
220
  }
199
221
  closePopup() {
222
+ const prevState = this.popupState;
200
223
  this.popupState = 'onHover';
224
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
201
225
  }
202
226
  togglePopup() {
203
227
  if (!this.errorMessageAsTooltip && !this.hasRightOfCaptionError()) {
@@ -206,12 +230,17 @@ export class FormElementComponent {
206
230
  if (this.errorFullyVisible) {
207
231
  return;
208
232
  }
233
+ const prevState = this.popupState;
209
234
  if (this.popupState === 'lockedOpen') {
210
235
  this.popupState = 'onHover';
211
236
  }
212
237
  else {
213
238
  this.popupState = 'lockedOpen';
214
239
  }
240
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
241
+ }
242
+ ngOnDestroy() {
243
+ this.subscriptions.forEach(e => e.unsubscribe());
215
244
  }
216
245
  }
217
246
  FormElementComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.8", ngImport: i0, type: FormElementComponent, deps: [{ token: i1.FormComponent, optional: true }, { token: FORM_ERROR_MESSAGES, optional: true }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
@@ -262,4 +291,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.8", ngImpor
262
291
  type: ContentChild,
263
292
  args: [NG_VALUE_ACCESSOR]
264
293
  }] } });
265
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-element.component.js","sourceRoot":"","sources":["../../../../../../../projects/klippa/ngx-enhancy-forms/src/lib/form/form-element/form-element.component.ts","../../../../../../../projects/klippa/ngx-enhancy-forms/src/lib/form/form-element/form-element.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n\tAfterViewInit,\n\tComponent,\n\tContentChild,\n\tElementRef,\n\tInject,\n\tInjectionToken,\n\tInput,\n\tOptional,\n\tTemplateRef,\n\tViewChild\n} from '@angular/core';\nimport {AbstractControl, NG_VALUE_ACCESSOR, UntypedFormControl} from '@angular/forms';\nimport {ValueAccessorBase} from '../../elements/value-accessor-base/value-accessor-base.component';\nimport {CustomErrorMessages, FormErrorMessages} from '../../types';\nimport {isValueSet, stringIsSetAndFilled} from '../../util/values';\nimport {FormComponent} from '../form.component';\nimport {awaitableForNextCycle} from '../../util/angular';\nimport {getAllLimitingContainers} from '../../util/dom';\n\n\nexport const FORM_ERROR_MESSAGES = new InjectionToken<CustomErrorMessages>('form.error.messages');\n\nexport const DEFAULT_ERROR_MESSAGES: FormErrorMessages = {\n\tmin: 'Use a number larger than %min%',\n\tmax: 'Use a number smaller than %max%',\n\trequired: 'Required',\n\temail: 'Use a valid email address',\n\tminLength: 'Has to be longer than %minLength% character(s)',\n\tmaxLength: 'Has to be shorter than %maxLength% character(s)',\n\tpattern: 'This input is not valid',\n\tmatchPassword: 'Passwords must match',\n\tdate: 'Enter a valid date',\n};\n\n@Component({\n\tselector: 'klp-form-element',\n\ttemplateUrl: './form-element.component.html',\n\tstyleUrls: ['./form-element.component.scss'],\n})\nexport class FormElementComponent implements AfterViewInit {\n\tpublic attachedControl: AbstractControl;\n\t@Input() public caption: string;\n\t@Input() public direction: 'horizontal' | 'vertical' = 'horizontal';\n\t@Input() public captionSpacing: 'percentages' | 'none' = 'percentages';\n\t@Input() public verticalAlignment: 'center' | 'top' = 'center';\n\t@Input() public spaceDistribution: '40-60' | '34-66' | '30-70' | 'fixedInputWidth' = '40-60';\n\t@Input() public swapInputAndCaption = false;\n\t@Input() public errorMessageAsTooltip = false;\n\t@ViewChild('internalComponentRef') public internalComponentRef: ElementRef;\n\t@ViewChild('tailTpl') public tailTpl: TemplateRef<any>;\n\t@ViewChild('captionDummyForSpaceCalculation') public captionDummyForSpaceCalculation: ElementRef;\n\t@ViewChild('absoluteAnchor') public absoluteAnchor: ElementRef;\n\t@ViewChild('fixedAnchor') public fixedAnchor: ElementRef;\n\t@ViewChild('fixedWrapper') public fixedWrapper: ElementRef;\n\t@ContentChild(NG_VALUE_ACCESSOR) fieldInput: ValueAccessorBase<any>;\n\n\tpublic captionRef: TemplateRef<any>;\n\tpublic errorMessages: FormErrorMessages = DEFAULT_ERROR_MESSAGES;\n\tpublic customErrorHandlers: Array<{ error: string; templateRef: TemplateRef<any> }> = [];\n\tprivate input: ValueAccessorBase<any>;\n\tpublic errorFullyVisible: boolean;\n\tprivate popupState: 'lockedOpen' | 'lockedClosed' | 'onHover' = 'onHover';\n\n\tconstructor(\n\t\t@Optional() private parent: FormComponent,\n\t\t@Inject(FORM_ERROR_MESSAGES) @Optional() private customMessages: CustomErrorMessages,\n\t\tprivate elRef: ElementRef,\n\t) {\n\t}\n\n\tasync ngAfterViewInit(): Promise<void> {\n\t\tawait awaitableForNextCycle();\n\t\tthis.fieldInput?.setTailTpl(this.tailTpl);\n\t\tthis.fieldInput?.onTouch.asObservable().subscribe((e) => {\n\t\t\tthis.determinePopupState();\n\t\t});\n\n\t\t[...getAllLimitingContainers(this.elRef.nativeElement), window].forEach(e => e.addEventListener('scroll', this.setErrorTooltipOffset));\n\t}\n\n\tpublic shouldShowErrorMessages(): boolean {\n\t\treturn this.parent?.showErrorMessages !== false;\n\t}\n\n\tpublic substituteParameters(message: string, parameters: Record<string, any>): string {\n\t\treturn Object.keys(parameters).reduce((msg, key) => {\n\t\t\treturn msg.replace(`%${key}%`, parameters[key]);\n\t\t}, message);\n\t}\n\n\tpublic registerControl(formControl: UntypedFormControl, input: ValueAccessorBase<any> = null): void {\n\t\tthis.attachedControl = formControl;\n\t\tthis.parent.registerControl(formControl, this);\n\t\tthis.input = input;\n\n\n\t\tthis.attachedControl.statusChanges.subscribe((e) => {\n\t\t\tthis.determinePopupState();\n\t\t});\n\t\tthis.determinePopupState();\n\t}\n\n\tpublic determinePopupState(): void {\n\t\tif (stringIsSetAndFilled(this.getErrorToShow())) {\n\t\t\tthis.popupState = 'onHover';\n\t\t\treturn;\n\t\t}\n\t\tif (isValueSet(this.getWarningToShow())) {\n\t\t\tthis.popupState = 'lockedOpen';\n\t\t\treturn;\n\t\t}\n\t\tthis.popupState = 'onHover';\n\t}\n\n\tpublic unregisterControl(formControl: UntypedFormControl): void {\n\t\tthis.attachedControl = null;\n\t\tthis.parent.unregisterControl(formControl);\n\t}\n\n\tpublic getAttachedControl(): AbstractControl {\n\t\treturn this.attachedControl;\n\t}\n\n\tpublic getAttachedInput(): ValueAccessorBase<any> {\n\t\treturn this.input;\n\t}\n\n\tpublic registerErrorHandler(error: string, templateRef: TemplateRef<any>): void {\n\t\tthis.customErrorHandlers.push({error, templateRef});\n\t}\n\n\tpublic registerCaption(templateRef: TemplateRef<any>): void {\n\t\tthis.captionRef = templateRef;\n\t}\n\n\tpublic getWarningToShow(): string | TemplateRef<any> {\n\t\treturn this.parent?.getWarningToShow(this.attachedControl);\n\t}\n\n\tpublic getWarningToShowAsTemplateRef(): TemplateRef<any> {\n\t\tif (this.parent?.getWarningToShow(this.attachedControl) instanceof TemplateRef) {\n\t\t\treturn this.parent?.getWarningToShow(this.attachedControl) as TemplateRef<any>;\n\t\t}\n\t\tthrow new Error('Warning is not a TemplateRef');\n\t}\n\n\tpublic getWarningToShowIsTemplateRef(): boolean {\n\t\treturn this.getWarningToShow() instanceof TemplateRef;\n\t}\n\n\tgetErrorToShow(): string {\n\t\tconst firstError = Object.keys(this.attachedControl?.errors ?? {})[0];\n\t\tif (this.attachedControl?.touched !== true) {\n\t\t\treturn null;\n\t\t}\n\t\tif (!this.attachedControl?.errors) {\n\t\t\treturn null;\n\t\t}\n\t\treturn firstError;\n\t}\n\n\tgetCustomErrorHandler(error: string): { error: string; templateRef: TemplateRef<any> } {\n\t\treturn this.customErrorHandlers.find((e) => e.error === error);\n\t}\n\n\tshowDefaultError(error: string): boolean {\n\t\treturn this.getErrorToShow() === error && !this.customErrorHandlers.some((e) => e.error === error);\n\t}\n\n\tgetScrollableParent(node): any {\n\t\tif (node === window.document.documentElement) {\n\t\t\treturn window.document.documentElement;\n\t\t}\n\t\tconst overflowY = getComputedStyle(node).overflowY;\n\t\tif (node.clientHeight < node.scrollHeight && (overflowY === 'auto' || overflowY === 'scroll')) {\n\t\t\treturn node;\n\t\t} else {\n\t\t\treturn this.getScrollableParent(node.parentNode);\n\t\t}\n\t}\n\n\tscrollTo(): void {\n\t\tconst parent = this.getScrollableParent(this.internalComponentRef.nativeElement);\n\t\tconst parentTop = parent === window.document.documentElement ? 0 : parent.getBoundingClientRect().top;\n\t\tconst elementTop = this.internalComponentRef.nativeElement.getBoundingClientRect().top;\n\t\tconst parentScrollTop = parent.scrollTop;\n\t\tconst answer = elementTop - parentTop + parentScrollTop;\n\n\t\tparent.scrollTo({\n\t\t\ttop: answer - 30,\n\t\t\tbehavior: 'smooth'\n\t\t});\n\t}\n\n\tisRequired(): boolean {\n\t\tif (isValueSet(this.input)) {\n\t\t\treturn this.input.hasValidator('required');\n\t\t}\n\t\treturn false;\n\t}\n\n\tgetErrorMessage(key: keyof FormErrorMessages): string {\n\t\tif (key === 'formLevel') {\n\t\t\treturn this.attachedControl.errors?.formLevel;\n\t\t}\n\t\treturn this.customMessages?.[key]?.() ?? this.errorMessages[key];\n\t}\n\n\tpublic getErrorLocation(): 'belowCaption' | 'rightOfCaption' {\n\t\treturn this.parent?.errorMessageLocation ?? 'belowCaption';\n\t}\n\n\tpublic shouldShowErrorTooltipOpened(): boolean {\n\t\treturn this.popupState === 'lockedOpen';\n\t}\n\n\tpublic hasHoverableErrorTooltip(): boolean {\n\t\tif (!this.hasRightOfCaptionError() && !this.errorMessageAsTooltip) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.popupState !== 'onHover') {\n\t\t\treturn false;\n\t\t}\n\t\tif (stringIsSetAndFilled(this.getErrorToShow())) {\n\t\t\treturn !this.errorFullyVisible;\n\t\t}\n\t\tif (isValueSet(this.getWarningToShow())) {\n\t\t\treturn true;\n\t\t}\n\t\treturn false;\n\t}\n\n\tpublic hasRightOfCaptionError(): boolean {\n\t\tif (this.errorMessageAsTooltip) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.direction !== 'vertical' || this.getErrorLocation() !== 'rightOfCaption') {\n\t\t\treturn false;\n\t\t}\n\t\treturn true;\n\t}\n\n\tpublic setErrorMessageIsTruncated = (isTruncated: boolean) => {\n\t\tthis.errorFullyVisible = !isTruncated;\n\t}\n\n\tpublic shouldShowWarningPopup(): boolean {\n\t\treturn isValueSet(this.getWarningToShow());\n\t}\n\n\tpublic closePopup(): void {\n\t\tthis.popupState = 'onHover';\n\t}\n\n\tpublic togglePopup(): void {\n\t\tif (!this.errorMessageAsTooltip && !this.hasRightOfCaptionError()) {\n\t\t\treturn;\n\t\t}\n\t\tif (this.errorFullyVisible) {\n\t\t\treturn;\n\t\t}\n\t\tif (this.popupState === 'lockedOpen') {\n\t\t\tthis.popupState = 'onHover';\n\t\t} else {\n\t\t\tthis.popupState = 'lockedOpen';\n\t\t}\n\t}\n\n\tpublic setErrorTooltipOffset = (): void => {\n\t\tif (this.popupState !== 'lockedOpen' && this.popupState !== 'onHover') {\n\t\t\treturn;\n\t\t}\n\t\tconst popupOffsetY = this.absoluteAnchor?.nativeElement.getBoundingClientRect().top - this.fixedAnchor?.nativeElement.getBoundingClientRect().top;\n\t\tif (this.fixedWrapper?.nativeElement) {\n\t\t\tthis.fixedWrapper.nativeElement.style.transform = `translateY(${popupOffsetY}px)`;\n\t\t}\n\t};\n}\n","<div\n\tclass=\"componentContainer\"\n\t[ngClass]=\"{\n\t\thasCaption: caption || captionRef,\n\t\tvertical: direction === 'vertical',\n\t\thorizontal: direction === 'horizontal',\n\t\ttopAlignment: verticalAlignment === 'top',\n\t\treverseOrder: swapInputAndCaption,\n\t\thasErrors: getErrorToShow() && attachedControl.touched,\n\t\tpercentageSpacing: captionSpacing === 'percentages' && spaceDistribution !== 'fixedInputWidth',\n\t\t'd40-60': spaceDistribution === '40-60',\n\t\t'd30-70': spaceDistribution === '30-70',\n\t\t'd34-66': spaceDistribution === '34-66',\n\t\t'fixedInputWidth': spaceDistribution === 'fixedInputWidth'\n\t}\"\n>\n\t<div class=\"errorAboveInputContainer\" *ngIf=\"direction === 'horizontal' && !errorMessageAsTooltip\">\n\t\t<div class=\"spacer\"></div>\n\t\t<ng-container [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t</div>\n\n\t<div class=\"captionInputAndError\" #internalComponentRef>\n\t\t<div class=\"captionDummyForSpaceCalculation\" #captionDummyForSpaceCalculation *ngIf=\"hasRightOfCaptionError()\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"captionTpl\" [ngTemplateOutletContext]=\"{forCalculation: true}\"></ng-container>\n\t\t</div>\n\t\t<ng-container [ngTemplateOutlet]=\"captionTpl\"></ng-container>\n\t\t<ng-container *ngIf=\"direction === 'vertical' && getErrorLocation() === 'belowCaption' && !errorMessageAsTooltip\" [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t<div class=\"inputContainer\" (mouseenter)=\"setErrorTooltipOffset()\">\n\t\t\t<ng-container *ngIf=\"errorMessageAsTooltip && shouldShowErrorMessages() && getErrorToShow()\">\n\t\t\t\t<div class=\"errorTooltipTriangle\"></div>\n\t\t\t\t<div class=\"errorTooltipTriangleWhite\"></div>\n\t\t\t\t<div class=\"errorTooltip\">\n\t\t\t\t\t<ng-container [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t\t\t</div>\n\t\t\t</ng-container>\n\t\t\t<ng-content></ng-content>\n\t\t</div>\n\t</div>\n</div>\n\n<ng-template #captionTpl let-forCalculation=\"forCalculation\">\n\t<div class=\"caption\"\n\t\t*ngIf=\"caption || captionRef\"\n\t\t[ngClass]=\"{\n\t\t\twithErrorRightOfCaption: getErrorLocation() === 'rightOfCaption'\n\t\t}\"\n\t>\n\t\t<div *ngIf=\"captionRef\" class=\"captionRefContainer\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"captionRef\"></ng-container>\n\t\t\t<div *ngIf=\"isRequired()\">&nbsp;*</div>\n\t\t</div>\n\t\t<div *ngIf=\"!captionRef\" class=\"captionText\">{{caption}}<span *ngIf=\"isRequired()\">&nbsp;*</span></div>\n\t\t<div class=\"rightOfCaptionError\" *ngIf=\"hasRightOfCaptionError()\" [ngClass]=\"{errorFullyVisible: errorFullyVisible}\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"errorRef\" [ngTemplateOutletContext]=\"{forCalculation: forCalculation}\"></ng-container>\n\t\t</div>\n\t</div>\n</ng-template>\n\n<ng-template #errorRef let-forCalculation=\"forCalculation\">\n\t<div *ngIf=\"shouldShowErrorMessages() && getErrorToShow()\" class=\"errorContainer\" [elementIsTruncatedCb]=\"forCalculation ? setErrorMessageIsTruncated : null\" [ngClass]=\"{horizontal: direction === 'horizontal', hasCaption: caption || captionRef, 'd30-70': spaceDistribution === '30-70', 'd34-66': spaceDistribution === '34-66'}\">\n\t\t<div *ngIf=\"showDefaultError('min')\">{{substituteParameters(getErrorMessage(\"min\"), {min: attachedControl.errors.min.min})}}</div>\n\t\t<div *ngIf=\"showDefaultError('max')\">{{substituteParameters(getErrorMessage(\"max\"), {max: attachedControl.errors.max.max})}}</div>\n\t\t<div *ngIf=\"showDefaultError('required')\">{{getErrorMessage(\"required\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('email')\">{{getErrorMessage(\"email\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('minlength')\">{{substituteParameters(getErrorMessage(\"minLength\"), {minLength: attachedControl.errors.minlength.requiredLength})}}</div>\n\t\t<div *ngIf=\"showDefaultError('maxlength')\">{{substituteParameters(getErrorMessage(\"maxLength\"), {maxLength: attachedControl.errors.maxlength.requiredLength})}}</div>\n\t\t<div *ngIf=\"showDefaultError('pattern')\">{{getErrorMessage(\"pattern\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('MatchPassword')\">{{getErrorMessage(\"matchPassword\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('date')\">{{getErrorMessage(\"date\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('message')\">{{attachedControl.errors.message.value}}</div>\n\t\t<div *ngIf=\"showDefaultError('formLevel')\">{{getErrorMessage(\"formLevel\")}}</div>\n\t\t<div [ngTemplateOutlet]=\"getCustomErrorHandler(getErrorToShow())?.templateRef\"></div>\n\t</div>\n</ng-template>\n\n<ng-template #tailTpl>\n\t<div class=\"errorTooltipContainer\" [ngClass]=\"{alwaysOpen: shouldShowErrorTooltipOpened()}\">\n\t\t<ng-container *ngIf=\"hasHoverableErrorTooltip() || shouldShowErrorTooltipOpened()\">\n\t\t\t<div class=\"errorTooltipTriangle\"></div>\n\t\t\t<div class=\"errorTooltipTriangleWhite\"></div>\n\n\t\t\t<div class=\"absoluteAnchor\" #absoluteAnchor></div>\n\t\t\t<div class=\"fixedAnchor\" #fixedAnchor [onRenderFn]=\"setErrorTooltipOffset\"></div>\n\t\t\t<div class=\"fixedWrapper\" #fixedWrapper>\n\t\t\t\t<div class=\"errorTooltip\" [ngClass]=\"{noPointerEvents: !shouldShowErrorTooltipOpened()}\">\n\t\t\t\t\t<div class=\"errorTooltipInner\">\n\t\t\t\t\t\t<i class=\"closeBtn\" (click)=\"closePopup();\">×</i>\n\t\t\t\t\t\t<ng-container *ngIf=\"getErrorToShow()\" [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t\t\t\t\t<div *ngIf=\"!getErrorToShow() && shouldShowWarningPopup()\">\n\t\t\t\t\t\t\t<ng-container *ngIf=\"getWarningToShowIsTemplateRef()\" [ngTemplateOutlet]=\"getWarningToShowAsTemplateRef()\"></ng-container>\n\t\t\t\t\t\t\t<span *ngIf=\"!getWarningToShowIsTemplateRef()\">{{getWarningToShow()}}</span>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t</ng-container>\n\t\t<klp-form-warning-icon variant=\"fill\" *ngIf=\"getErrorToShow()\" (click)=\"togglePopup()\"></klp-form-warning-icon>\n\t\t<klp-form-warning-icon variant=\"line\" *ngIf=\"!getErrorToShow() && getWarningToShow()\" (click)=\"togglePopup()\"></klp-form-warning-icon>\n\t</div>\n</ng-template>\n"]}
294
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-element.component.js","sourceRoot":"","sources":["../../../../../../../projects/klippa/ngx-enhancy-forms/src/lib/form/form-element/form-element.component.ts","../../../../../../../projects/klippa/ngx-enhancy-forms/src/lib/form/form-element/form-element.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n\tAfterViewInit,\n\tComponent,\n\tContentChild,\n\tElementRef,\n\tInject,\n\tInjectionToken,\n\tInput, OnDestroy,\n\tOptional,\n\tTemplateRef,\n\tViewChild\n} from '@angular/core';\nimport {AbstractControl, NG_VALUE_ACCESSOR, UntypedFormControl} from '@angular/forms';\nimport {ValueAccessorBase} from '../../elements/value-accessor-base/value-accessor-base.component';\nimport {CustomErrorMessages, FormErrorMessages} from '../../types';\nimport {isValueSet, stringIsSetAndFilled} from '../../util/values';\nimport {FormComponent} from '../form.component';\nimport {awaitableForNextCycle} from '../../util/angular';\nimport {getAllLimitingContainers} from '../../util/dom';\nimport {Subscription} from \"rxjs\";\n\n\nexport const FORM_ERROR_MESSAGES = new InjectionToken<CustomErrorMessages>('form.error.messages');\n\nexport const DEFAULT_ERROR_MESSAGES: FormErrorMessages = {\n\tmin: 'Use a number larger than %min%',\n\tmax: 'Use a number smaller than %max%',\n\trequired: 'Required',\n\temail: 'Use a valid email address',\n\tminLength: 'Has to be longer than %minLength% character(s)',\n\tmaxLength: 'Has to be shorter than %maxLength% character(s)',\n\tpattern: 'This input is not valid',\n\tmatchPassword: 'Passwords must match',\n\tdate: 'Enter a valid date',\n};\n\ntype PopupState = 'onHover' | 'lockedOpen';\n\n@Component({\n\tselector: 'klp-form-element',\n\ttemplateUrl: './form-element.component.html',\n\tstyleUrls: ['./form-element.component.scss'],\n})\nexport class FormElementComponent implements AfterViewInit, OnDestroy {\n\tpublic attachedControl: AbstractControl;\n\t@Input() public caption: string;\n\t@Input() public direction: 'horizontal' | 'vertical' = 'horizontal';\n\t@Input() public captionSpacing: 'percentages' | 'none' = 'percentages';\n\t@Input() public verticalAlignment: 'center' | 'top' = 'center';\n\t@Input() public spaceDistribution: '40-60' | '34-66' | '30-70' | 'fixedInputWidth' = '40-60';\n\t@Input() public swapInputAndCaption = false;\n\t@Input() public errorMessageAsTooltip = false;\n\t@ViewChild('internalComponentRef') public internalComponentRef: ElementRef;\n\t@ViewChild('tailTpl') public tailTpl: TemplateRef<any>;\n\t@ViewChild('captionDummyForSpaceCalculation') public captionDummyForSpaceCalculation: ElementRef;\n\t@ViewChild('absoluteAnchor') public absoluteAnchor: ElementRef;\n\t@ViewChild('fixedAnchor') public fixedAnchor: ElementRef;\n\t@ViewChild('fixedWrapper') public fixedWrapper: ElementRef;\n\t@ContentChild(NG_VALUE_ACCESSOR) fieldInput: ValueAccessorBase<any>;\n\n\tpublic captionRef: TemplateRef<any>;\n\tpublic errorMessages: FormErrorMessages = DEFAULT_ERROR_MESSAGES;\n\tpublic customErrorHandlers: Array<{ error: string; templateRef: TemplateRef<any> }> = [];\n\tprivate input: ValueAccessorBase<any>;\n\tpublic errorFullyVisible: boolean;\n\tprivate popupState: PopupState = 'onHover';\n\tprivate subscriptions: Array<Subscription> = [];\n\n\tconstructor(\n\t\t@Optional() private parent: FormComponent,\n\t\t@Inject(FORM_ERROR_MESSAGES) @Optional() private customMessages: CustomErrorMessages,\n\t\tprivate elRef: ElementRef,\n\t) {\n\t}\n\n\tasync ngAfterViewInit(): Promise<void> {\n\t\tawait awaitableForNextCycle();\n\t\tthis.fieldInput?.setTailTpl(this.tailTpl);\n\t\tconst subscription = this.fieldInput?.onTouch.asObservable().subscribe(() => {\n\t\t\tthis.determinePopupState();\n\t\t});\n\t\tif (isValueSet(subscription))  {\n\t\t\tthis.subscriptions.push(subscription);\n\t\t}\n\t}\n\n\tpublic shouldShowErrorMessages(): boolean {\n\t\treturn this.parent?.showErrorMessages !== false;\n\t}\n\n\tpublic substituteParameters(message: string, parameters: Record<string, any>): string {\n\t\treturn Object.keys(parameters).reduce((msg, key) => {\n\t\t\treturn msg.replace(`%${key}%`, parameters[key]);\n\t\t}, message);\n\t}\n\n\tpublic registerControl(formControl: UntypedFormControl, input: ValueAccessorBase<any> = null): void {\n\t\tthis.attachedControl = formControl;\n\t\tthis.parent.registerControl(formControl, this);\n\t\tthis.input = input;\n\n\n\t\tconst subscription = this.attachedControl.statusChanges.subscribe(() => {\n\t\t\tthis.determinePopupState();\n\t\t});\n\t\tthis.subscriptions.push(subscription);\n\t\tthis.determinePopupState();\n\t}\n\n\tpublic determinePopupState(): void {\n\t\tconst prevState = this.popupState;\n\t\tif (stringIsSetAndFilled(this.getErrorToShow())) {\n\t\t\tthis.popupState = 'onHover';\n\t\t} else if (isValueSet(this.getWarningToShow())) {\n\t\t\tthis.popupState = 'lockedOpen';\n\t\t} else {\n\t\t\tthis.popupState = 'onHover';\n\t\t}\n\n\t\tthis.setUpErrorTooltipListeners(prevState, this.popupState);\n\t}\n\n\tprivate setUpErrorTooltipListeners(prev: PopupState, current: PopupState): void {\n\t\tif (prev === current) {\n\t\t\treturn;\n\t\t}\n\t\tconst containers = [...getAllLimitingContainers(this.elRef.nativeElement), window];\n\t\tif (current === 'lockedOpen') {\n\t\t\tcontainers.forEach(e => {\n\t\t\t\te.addEventListener('scroll', this.setErrorTooltipOffset);\n\t\t\t});\n\t\t} else {\n\t\t\tcontainers.forEach(e => {\n\t\t\t\te.removeEventListener('scroll', this.setErrorTooltipOffset);\n\t\t\t});\n\t\t}\n\t}\n\n\tpublic unregisterControl(formControl: UntypedFormControl): void {\n\t\tthis.attachedControl = null;\n\t\tthis.parent.unregisterControl(formControl);\n\t}\n\n\tpublic getAttachedControl(): AbstractControl {\n\t\treturn this.attachedControl;\n\t}\n\n\tpublic getAttachedInput(): ValueAccessorBase<any> {\n\t\treturn this.input;\n\t}\n\n\tpublic registerErrorHandler(error: string, templateRef: TemplateRef<any>): void {\n\t\tthis.customErrorHandlers.push({error, templateRef});\n\t}\n\n\tpublic registerCaption(templateRef: TemplateRef<any>): void {\n\t\tthis.captionRef = templateRef;\n\t}\n\n\tpublic getWarningToShow(): string | TemplateRef<any> {\n\t\treturn this.parent?.getWarningToShow(this.attachedControl);\n\t}\n\n\tpublic getWarningToShowAsTemplateRef(): TemplateRef<any> {\n\t\tif (this.parent?.getWarningToShow(this.attachedControl) instanceof TemplateRef) {\n\t\t\treturn this.parent?.getWarningToShow(this.attachedControl) as TemplateRef<any>;\n\t\t}\n\t\tthrow new Error('Warning is not a TemplateRef');\n\t}\n\n\tpublic getWarningToShowIsTemplateRef(): boolean {\n\t\treturn this.getWarningToShow() instanceof TemplateRef;\n\t}\n\n\tgetErrorToShow(): string {\n\t\tconst firstError = Object.keys(this.attachedControl?.errors ?? {})[0];\n\t\tif (this.attachedControl?.touched !== true) {\n\t\t\treturn null;\n\t\t}\n\t\tif (!this.attachedControl?.errors) {\n\t\t\treturn null;\n\t\t}\n\t\treturn firstError;\n\t}\n\n\tgetCustomErrorHandler(error: string): { error: string; templateRef: TemplateRef<any> } {\n\t\treturn this.customErrorHandlers.find((e) => e.error === error);\n\t}\n\n\tshowDefaultError(error: string): boolean {\n\t\treturn this.getErrorToShow() === error && !this.customErrorHandlers.some((e) => e.error === error);\n\t}\n\n\tgetScrollableParent(node): any {\n\t\tif (node === window.document.documentElement) {\n\t\t\treturn window.document.documentElement;\n\t\t}\n\t\tconst overflowY = getComputedStyle(node).overflowY;\n\t\tif (node.clientHeight < node.scrollHeight && (overflowY === 'auto' || overflowY === 'scroll')) {\n\t\t\treturn node;\n\t\t} else {\n\t\t\treturn this.getScrollableParent(node.parentNode);\n\t\t}\n\t}\n\n\tscrollTo(): void {\n\t\tconst parent = this.getScrollableParent(this.internalComponentRef.nativeElement);\n\t\tconst parentTop = parent === window.document.documentElement ? 0 : parent.getBoundingClientRect().top;\n\t\tconst elementTop = this.internalComponentRef.nativeElement.getBoundingClientRect().top;\n\t\tconst parentScrollTop = parent.scrollTop;\n\t\tconst answer = elementTop - parentTop + parentScrollTop;\n\n\t\tparent.scrollTo({\n\t\t\ttop: answer - 30,\n\t\t\tbehavior: 'smooth'\n\t\t});\n\t}\n\n\tisRequired(): boolean {\n\t\tif (isValueSet(this.input)) {\n\t\t\treturn this.input.hasValidator('required');\n\t\t}\n\t\treturn false;\n\t}\n\n\tgetErrorMessage(key: keyof FormErrorMessages): string {\n\t\tif (key === 'formLevel') {\n\t\t\treturn this.attachedControl.errors?.formLevel;\n\t\t}\n\t\treturn this.customMessages?.[key]?.() ?? this.errorMessages[key];\n\t}\n\n\tpublic getErrorLocation(): 'belowCaption' | 'rightOfCaption' {\n\t\treturn this.parent?.errorMessageLocation ?? 'belowCaption';\n\t}\n\n\tpublic shouldShowErrorTooltipOpened(): boolean {\n\t\treturn this.popupState === 'lockedOpen';\n\t}\n\n\tpublic hasHoverableErrorTooltip(): boolean {\n\t\tif (!this.hasRightOfCaptionError() && !this.errorMessageAsTooltip) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.popupState !== 'onHover') {\n\t\t\treturn false;\n\t\t}\n\t\tif (stringIsSetAndFilled(this.getErrorToShow())) {\n\t\t\treturn !this.errorFullyVisible;\n\t\t}\n\t\tif (isValueSet(this.getWarningToShow())) {\n\t\t\treturn true;\n\t\t}\n\t\treturn false;\n\t}\n\n\tpublic hasRightOfCaptionError(): boolean {\n\t\tif (this.errorMessageAsTooltip) {\n\t\t\treturn false;\n\t\t}\n\t\tif (this.direction !== 'vertical' || this.getErrorLocation() !== 'rightOfCaption') {\n\t\t\treturn false;\n\t\t}\n\t\treturn true;\n\t}\n\n\tpublic setErrorMessageIsTruncated = (isTruncated: boolean) => {\n\t\tthis.errorFullyVisible = !isTruncated;\n\t}\n\n\tpublic shouldShowWarningPopup(): boolean {\n\t\treturn isValueSet(this.getWarningToShow());\n\t}\n\n\tpublic closePopup(): void {\n\t\tconst prevState = this.popupState;\n\t\tthis.popupState = 'onHover';\n\t\tthis.setUpErrorTooltipListeners(prevState, this.popupState);\n\t}\n\n\tpublic togglePopup(): void {\n\t\tif (!this.errorMessageAsTooltip && !this.hasRightOfCaptionError()) {\n\t\t\treturn;\n\t\t}\n\t\tif (this.errorFullyVisible) {\n\t\t\treturn;\n\t\t}\n\t\tconst prevState = this.popupState;\n\t\tif (this.popupState === 'lockedOpen') {\n\t\t\tthis.popupState = 'onHover';\n\t\t} else {\n\t\t\tthis.popupState = 'lockedOpen';\n\t\t}\n\t\tthis.setUpErrorTooltipListeners(prevState, this.popupState);\n\t}\n\n\tpublic setErrorTooltipOffset = (): void => {\n\t\tif (this.popupState !== 'lockedOpen' && this.popupState !== 'onHover') {\n\t\t\treturn;\n\t\t}\n\t\tconst popupOffsetY = this.absoluteAnchor?.nativeElement.getBoundingClientRect().top - this.fixedAnchor?.nativeElement.getBoundingClientRect().top;\n\t\tif (this.fixedWrapper?.nativeElement) {\n\t\t\tthis.fixedWrapper.nativeElement.style.transform = `translateY(${popupOffsetY}px)`;\n\t\t}\n\t};\n\n\tngOnDestroy(): void {\n\t\tthis.subscriptions.forEach(e => e.unsubscribe());\n\t}\n}\n","<div\n\tclass=\"componentContainer\"\n\t[ngClass]=\"{\n\t\thasCaption: caption || captionRef,\n\t\tvertical: direction === 'vertical',\n\t\thorizontal: direction === 'horizontal',\n\t\ttopAlignment: verticalAlignment === 'top',\n\t\treverseOrder: swapInputAndCaption,\n\t\thasErrors: getErrorToShow() && attachedControl.touched,\n\t\tpercentageSpacing: captionSpacing === 'percentages' && spaceDistribution !== 'fixedInputWidth',\n\t\t'd40-60': spaceDistribution === '40-60',\n\t\t'd30-70': spaceDistribution === '30-70',\n\t\t'd34-66': spaceDistribution === '34-66',\n\t\t'fixedInputWidth': spaceDistribution === 'fixedInputWidth'\n\t}\"\n>\n\t<div class=\"errorAboveInputContainer\" *ngIf=\"direction === 'horizontal' && !errorMessageAsTooltip\">\n\t\t<div class=\"spacer\"></div>\n\t\t<ng-container [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t</div>\n\n\t<div class=\"captionInputAndError\" #internalComponentRef>\n\t\t<div class=\"captionDummyForSpaceCalculation\" #captionDummyForSpaceCalculation *ngIf=\"hasRightOfCaptionError()\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"captionTpl\" [ngTemplateOutletContext]=\"{forCalculation: true}\"></ng-container>\n\t\t</div>\n\t\t<ng-container [ngTemplateOutlet]=\"captionTpl\"></ng-container>\n\t\t<ng-container *ngIf=\"direction === 'vertical' && getErrorLocation() === 'belowCaption' && !errorMessageAsTooltip\" [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t<div class=\"inputContainer\" (mouseenter)=\"setErrorTooltipOffset()\">\n\t\t\t<ng-container *ngIf=\"errorMessageAsTooltip && shouldShowErrorMessages() && getErrorToShow()\">\n\t\t\t\t<div class=\"errorTooltipTriangle\"></div>\n\t\t\t\t<div class=\"errorTooltipTriangleWhite\"></div>\n\t\t\t\t<div class=\"errorTooltip\">\n\t\t\t\t\t<ng-container [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t\t\t</div>\n\t\t\t</ng-container>\n\t\t\t<ng-content></ng-content>\n\t\t</div>\n\t</div>\n</div>\n\n<ng-template #captionTpl let-forCalculation=\"forCalculation\">\n\t<div class=\"caption\"\n\t\t*ngIf=\"caption || captionRef\"\n\t\t[ngClass]=\"{\n\t\t\twithErrorRightOfCaption: getErrorLocation() === 'rightOfCaption'\n\t\t}\"\n\t>\n\t\t<div *ngIf=\"captionRef\" class=\"captionRefContainer\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"captionRef\"></ng-container>\n\t\t\t<div *ngIf=\"isRequired()\">&nbsp;*</div>\n\t\t</div>\n\t\t<div *ngIf=\"!captionRef\" class=\"captionText\">{{caption}}<span *ngIf=\"isRequired()\">&nbsp;*</span></div>\n\t\t<div class=\"rightOfCaptionError\" *ngIf=\"hasRightOfCaptionError()\" [ngClass]=\"{errorFullyVisible: errorFullyVisible}\">\n\t\t\t<ng-container [ngTemplateOutlet]=\"errorRef\" [ngTemplateOutletContext]=\"{forCalculation: forCalculation}\"></ng-container>\n\t\t</div>\n\t</div>\n</ng-template>\n\n<ng-template #errorRef let-forCalculation=\"forCalculation\">\n\t<div *ngIf=\"shouldShowErrorMessages() && getErrorToShow()\" class=\"errorContainer\" [elementIsTruncatedCb]=\"forCalculation ? setErrorMessageIsTruncated : null\" [ngClass]=\"{horizontal: direction === 'horizontal', hasCaption: caption || captionRef, 'd30-70': spaceDistribution === '30-70', 'd34-66': spaceDistribution === '34-66'}\">\n\t\t<div *ngIf=\"showDefaultError('min')\">{{substituteParameters(getErrorMessage(\"min\"), {min: attachedControl.errors.min.min})}}</div>\n\t\t<div *ngIf=\"showDefaultError('max')\">{{substituteParameters(getErrorMessage(\"max\"), {max: attachedControl.errors.max.max})}}</div>\n\t\t<div *ngIf=\"showDefaultError('required')\">{{getErrorMessage(\"required\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('email')\">{{getErrorMessage(\"email\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('minlength')\">{{substituteParameters(getErrorMessage(\"minLength\"), {minLength: attachedControl.errors.minlength.requiredLength})}}</div>\n\t\t<div *ngIf=\"showDefaultError('maxlength')\">{{substituteParameters(getErrorMessage(\"maxLength\"), {maxLength: attachedControl.errors.maxlength.requiredLength})}}</div>\n\t\t<div *ngIf=\"showDefaultError('pattern')\">{{getErrorMessage(\"pattern\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('MatchPassword')\">{{getErrorMessage(\"matchPassword\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('date')\">{{getErrorMessage(\"date\")}}</div>\n\t\t<div *ngIf=\"showDefaultError('message')\">{{attachedControl.errors.message.value}}</div>\n\t\t<div *ngIf=\"showDefaultError('formLevel')\">{{getErrorMessage(\"formLevel\")}}</div>\n\t\t<div [ngTemplateOutlet]=\"getCustomErrorHandler(getErrorToShow())?.templateRef\"></div>\n\t</div>\n</ng-template>\n\n<ng-template #tailTpl>\n\t<div class=\"errorTooltipContainer\" [ngClass]=\"{alwaysOpen: shouldShowErrorTooltipOpened()}\">\n\t\t<ng-container *ngIf=\"hasHoverableErrorTooltip() || shouldShowErrorTooltipOpened()\">\n\t\t\t<div class=\"errorTooltipTriangle\"></div>\n\t\t\t<div class=\"errorTooltipTriangleWhite\"></div>\n\n\t\t\t<div class=\"absoluteAnchor\" #absoluteAnchor></div>\n\t\t\t<div class=\"fixedAnchor\" #fixedAnchor [onRenderFn]=\"setErrorTooltipOffset\"></div>\n\t\t\t<div class=\"fixedWrapper\" #fixedWrapper>\n\t\t\t\t<div class=\"errorTooltip\" [ngClass]=\"{noPointerEvents: !shouldShowErrorTooltipOpened()}\">\n\t\t\t\t\t<div class=\"errorTooltipInner\">\n\t\t\t\t\t\t<i class=\"closeBtn\" (click)=\"closePopup();\">×</i>\n\t\t\t\t\t\t<ng-container *ngIf=\"getErrorToShow()\" [ngTemplateOutlet]=\"errorRef\"></ng-container>\n\t\t\t\t\t\t<div *ngIf=\"!getErrorToShow() && shouldShowWarningPopup()\">\n\t\t\t\t\t\t\t<ng-container *ngIf=\"getWarningToShowIsTemplateRef()\" [ngTemplateOutlet]=\"getWarningToShowAsTemplateRef()\"></ng-container>\n\t\t\t\t\t\t\t<span *ngIf=\"!getWarningToShowIsTemplateRef()\">{{getWarningToShow()}}</span>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t</ng-container>\n\t\t<klp-form-warning-icon variant=\"fill\" *ngIf=\"getErrorToShow()\" (click)=\"togglePopup()\"></klp-form-warning-icon>\n\t\t<klp-form-warning-icon variant=\"line\" *ngIf=\"!getErrorToShow() && getWarningToShow()\" (click)=\"togglePopup()\"></klp-form-warning-icon>\n\t</div>\n</ng-template>\n"]}
@@ -588,6 +588,7 @@ class FormElementComponent {
588
588
  this.errorMessages = DEFAULT_ERROR_MESSAGES;
589
589
  this.customErrorHandlers = [];
590
590
  this.popupState = 'onHover';
591
+ this.subscriptions = [];
591
592
  this.setErrorMessageIsTruncated = (isTruncated) => {
592
593
  this.errorFullyVisible = !isTruncated;
593
594
  };
@@ -607,10 +608,12 @@ class FormElementComponent {
607
608
  return __awaiter(this, void 0, void 0, function* () {
608
609
  yield awaitableForNextCycle();
609
610
  (_a = this.fieldInput) === null || _a === void 0 ? void 0 : _a.setTailTpl(this.tailTpl);
610
- (_b = this.fieldInput) === null || _b === void 0 ? void 0 : _b.onTouch.asObservable().subscribe((e) => {
611
+ const subscription = (_b = this.fieldInput) === null || _b === void 0 ? void 0 : _b.onTouch.asObservable().subscribe(() => {
611
612
  this.determinePopupState();
612
613
  });
613
- [...getAllLimitingContainers(this.elRef.nativeElement), window].forEach(e => e.addEventListener('scroll', this.setErrorTooltipOffset));
614
+ if (isValueSet(subscription)) {
615
+ this.subscriptions.push(subscription);
616
+ }
614
617
  });
615
618
  }
616
619
  shouldShowErrorMessages() {
@@ -626,21 +629,40 @@ class FormElementComponent {
626
629
  this.attachedControl = formControl;
627
630
  this.parent.registerControl(formControl, this);
628
631
  this.input = input;
629
- this.attachedControl.statusChanges.subscribe((e) => {
632
+ const subscription = this.attachedControl.statusChanges.subscribe(() => {
630
633
  this.determinePopupState();
631
634
  });
635
+ this.subscriptions.push(subscription);
632
636
  this.determinePopupState();
633
637
  }
634
638
  determinePopupState() {
639
+ const prevState = this.popupState;
635
640
  if (stringIsSetAndFilled(this.getErrorToShow())) {
636
641
  this.popupState = 'onHover';
637
- return;
638
642
  }
639
- if (isValueSet(this.getWarningToShow())) {
643
+ else if (isValueSet(this.getWarningToShow())) {
640
644
  this.popupState = 'lockedOpen';
645
+ }
646
+ else {
647
+ this.popupState = 'onHover';
648
+ }
649
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
650
+ }
651
+ setUpErrorTooltipListeners(prev, current) {
652
+ if (prev === current) {
641
653
  return;
642
654
  }
643
- this.popupState = 'onHover';
655
+ const containers = [...getAllLimitingContainers(this.elRef.nativeElement), window];
656
+ if (current === 'lockedOpen') {
657
+ containers.forEach(e => {
658
+ e.addEventListener('scroll', this.setErrorTooltipOffset);
659
+ });
660
+ }
661
+ else {
662
+ containers.forEach(e => {
663
+ e.removeEventListener('scroll', this.setErrorTooltipOffset);
664
+ });
665
+ }
644
666
  }
645
667
  unregisterControl(formControl) {
646
668
  this.attachedControl = null;
@@ -760,7 +782,9 @@ class FormElementComponent {
760
782
  return isValueSet(this.getWarningToShow());
761
783
  }
762
784
  closePopup() {
785
+ const prevState = this.popupState;
763
786
  this.popupState = 'onHover';
787
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
764
788
  }
765
789
  togglePopup() {
766
790
  if (!this.errorMessageAsTooltip && !this.hasRightOfCaptionError()) {
@@ -769,12 +793,17 @@ class FormElementComponent {
769
793
  if (this.errorFullyVisible) {
770
794
  return;
771
795
  }
796
+ const prevState = this.popupState;
772
797
  if (this.popupState === 'lockedOpen') {
773
798
  this.popupState = 'onHover';
774
799
  }
775
800
  else {
776
801
  this.popupState = 'lockedOpen';
777
802
  }
803
+ this.setUpErrorTooltipListeners(prevState, this.popupState);
804
+ }
805
+ ngOnDestroy() {
806
+ this.subscriptions.forEach(e => e.unsubscribe());
778
807
  }
779
808
  }
780
809
  FormElementComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.8", ngImport: i0, type: FormElementComponent, deps: [{ token: FormComponent, optional: true }, { token: FORM_ERROR_MESSAGES, optional: true }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });