UNPKG

kwikui

Version:

KwikID's UI Component Library in Angular

272 lines 39.9 kB
import { Component, EventEmitter, forwardRef, Input, Output, ViewChild } from "@angular/core"; import { FormControl, FormGroup, NG_VALUE_ACCESSOR } from "@angular/forms"; import { Subject, Subscription } from "rxjs"; import { debounceTime, distinctUntilChanged } from "rxjs/operators"; import { throwErrorMessage } from "../../../helpers/kwikui.common.helpers"; import { DEFAULT_VALUES, VALIDATE_KEY_VALUES } from "./input-email.constants"; import { isValidKeyValue } from "./input-email.validation"; import * as i0 from "@angular/core"; import * as i1 from "@taiga-ui/kit"; import * as i2 from "@taiga-ui/core"; import * as i3 from "@angular/forms"; import * as i4 from "@angular/common"; export class KwikUIInputEmailComponent { constructor(cdr) { this.cdr = cdr; this.case = DEFAULT_VALUES.case; this.disabled = DEFAULT_VALUES.disabled; this.domains = DEFAULT_VALUES.domains; this.focus = DEFAULT_VALUES.focus; this.formControl = new FormControl({}); this.formControlName = DEFAULT_VALUES.formControlName; this.hintContent = DEFAULT_VALUES.hintContent; this.icon = DEFAULT_VALUES.icon; this.id = DEFAULT_VALUES.id; this.invalid = DEFAULT_VALUES.invalid; this.label = DEFAULT_VALUES.label; this.placeholder = DEFAULT_VALUES.placeholder; this.postfix = DEFAULT_VALUES.postfix; this.prefix = DEFAULT_VALUES.prefix; this.properties = { readOnly: false }; this.size = DEFAULT_VALUES.size; this.validators = { required: true }; this.getKeyValue = new EventEmitter(); this.dropdownPosition = undefined; this.suggestions = []; this.domainSelected = undefined; this.formGroup = new FormGroup({}); this.subscriptions = new Subscription(); this.inputSubject = new Subject(); } ngOnInit() { this.formGroup.addControl(this.formControlName, this.formControl); this.focus = this.focus === undefined ? false : this.focus; this.invalid = this.invalid === undefined ? false : this.invalid && this.focus; this.validators = Object.assign({}, this.validators); this.properties = Object.assign({}, this.properties); this.changeCase(); this.setDisabled(); this.subscriptions.add(this.formGroup.controls[this.formControlName].valueChanges .pipe(distinctUntilChanged()) // makes sure the value has actually changed. .subscribe((value) => this.handleInputValueChange(value))); this.inputSubject.pipe(debounceTime(300)).subscribe(() => { this.updateSuggestions(); }); } ngOnChanges(changes) { const verifyChange = (key) => { return changes.hasOwnProperty(key) && !changes[key].firstChange; }; for (const change of Object.entries(changes)) { const key = change[0]; const value = change[1].currentValue; this.validateInputProperty(key, value); } if (verifyChange("formControl")) { this.formControl = changes.formControl.currentValue; } if (verifyChange("invalid")) { this.invalid = Boolean(changes.invalid.currentValue); this.setError(); } if (verifyChange("focus")) { this.focus = Boolean(changes.focus.currentValue); } if (verifyChange("disabled")) { this.disabled = Boolean(changes.disabled.currentValue); this.setDisabled(); } } ngOnDestroy() { this.subscriptions.unsubscribe(); // ensure when component is destroyed the subscription is also and not left open. } validateInputProperty(key, value) { if (VALIDATE_KEY_VALUES[key] && !isValidKeyValue(key, value)) { this[key] = DEFAULT_VALUES[key]; throwErrorMessage("kwikui-input-email", this.id, key, value, DEFAULT_VALUES[key]); } } /** * @description Handles setting up of error and focus on the input field is it is invalid */ setDisabled() { if (this.formGroup.controls[this.formControlName] !== undefined) { if (this.disabled === true) { this.formGroup.controls[this.formControlName].disable({ emitEvent: false }); } else { this.formGroup.controls[this.formControlName].enable({ emitEvent: false }); } } } /** * @description Handles setting up of error and focus on the input field is it is invalid */ setError() { if (this.formGroup.controls[this.formControlName] !== undefined && this.formGroup.controls[this.formControlName].invalid) { this.invalid = true; this.focus = true; } else { this.invalid = false; this.focus = false; } } /** * @description Handle changing of case based on 'case' props */ changeCase() { switch (this.case) { case "upper": this.formGroup.patchValue({ [this.formControlName]: this.formGroup.controls[this.formControlName].value.toLocaleUpperCase() }); break; case "lower": this.formGroup.patchValue({ [this.formControlName]: this.formGroup.controls[this.formControlName].value.toLocaleLowerCase() }); break; default: break; } } /** * @description Handles firing of 2 events on (keyup) event * * @param value */ handleInputValueChange(value) { this.changeCase(); this.inputSubject.next(this.formControl.value); if (value === this.formControl.value) { this.emitEvent(this.getKeyValue, { key: this.formControlName, value: this.formControl.value }); } } /** * @description Update the suggestion domains */ updateSuggestions() { const atIndex = this.formControl.value.lastIndexOf("@"); this.suggestions = []; if (!this.domainSelected && atIndex != -1) { const domainInput = this.formControl.value.substring(atIndex + 1); this.suggestions = this.domains.filter((domain) => domain.startsWith(domainInput)); this.cdr.detectChanges(); this.checkDropdownPosition(); } this.domainSelected = undefined; } /** * @description Handles Position of Suggestion Dropdown */ checkDropdownPosition() { if (this.dropdown) { const rectRelative = this.inputEmail.nativeElement.getBoundingClientRect(); const rectAbsolute = this.dropdown.nativeElement.getBoundingClientRect(); if (rectAbsolute.bottom + rectRelative.height > window.innerHeight) { this.dropdownPosition = `-${rectAbsolute.height + 8}px`; } else { this.dropdownPosition = "100%"; } } } /** * @description Handle selection of domains */ selectDomain(domain) { this.formControl.setValue(this.formControl.value.replace(/@[\w.-]*$/, `@${domain}`)); this.suggestions = []; this.domainSelected = domain; } handleFocusedChange(e) { if (!this.invalid) { this.focus = e; } } emitEvent(event, data) { event.emit(data); } /** Method Implementations for Abstract Control */ writeValue(value) { } registerOnChange(fn) { } registerOnTouched(fn) { } setDisabledState(isDisabled) { } } /** @nocollapse */ KwikUIInputEmailComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: KwikUIInputEmailComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ KwikUIInputEmailComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.17", type: KwikUIInputEmailComponent, selector: "kwikui-input-email", inputs: { case: "case", disabled: "disabled", domains: "domains", focus: "focus", formControl: "formControl", formControlName: "formControlName", hintContent: "hintContent", icon: "icon", id: "id", invalid: "invalid", label: "label", placeholder: "placeholder", postfix: "postfix", prefix: "prefix", properties: "properties", size: "size", validators: "validators" }, outputs: { getKeyValue: "getKeyValue" }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef((() => KwikUIInputEmailComponent)), multi: true } ], viewQueries: [{ propertyName: "inputEmail", first: true, predicate: ["inputEmail"], descendants: true }, { propertyName: "dropdown", first: true, predicate: ["dropdown"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n id=\"input-email\"\n [formGroup]=\"formGroup\"\n #inputEmail\n>\n <tui-input\n [formControlName]=\"formControlName\"\n [nativeId]=\"id\"\n [pseudoFocus]=\"focus ?? null\"\n [pseudoInvalid]=\"invalid === undefined ? null : invalid\"\n [readOnly]=\"properties?.readOnly ?? false\"\n [tuiHintContent]=\"hintContent\"\n [tuiTextfieldCleaner]=\"true\"\n [tuiTextfieldIconLeft]=\"icon\"\n [tuiTextfieldPostfix]=\"postfix\"\n [tuiTextfieldPrefix]=\"prefix\"\n [tuiTextfieldSize]=\"size\"\n (focusedChange)=\"handleFocusedChange($event)\"\n >\n {{ label }}\n <span\n class=\"tui-required\"\n *ngIf=\"validators.required\"\n ></span>\n <input\n tuiTextfield\n [attr.type]=\"'email'\"\n [attr.placeholder]=\"placeholder ?? ''\"\n [attr.required]=\"validators.required\"\n />\n </tui-input>\n <div\n #dropdown\n [class.input-email-s]=\"size === 's'\"\n [class.input-email-m]=\"size === 'm'\"\n [class.input-email-l]=\"size === 'l'\"\n [style.top]=\"dropdownPosition\"\n id=\"suggestions\"\n *ngIf=\"suggestions && suggestions.length > 0\"\n >\n <div id=\"suggestions-container\"\n ><div\n class=\"suggestion\"\n *ngFor=\"let domain of suggestions\"\n (click)=\"selectDomain(domain)\"\n >{{ domain }}</div\n ></div\n >\n </div>\n</div>\n", styles: ["::-webkit-scrollbar{width:4px;height:4px}::-webkit-scrollbar-track{border-radius:4px}::-webkit-scrollbar-thumb{background:lightgray;border-radius:4px}::-webkit-scrollbar-thumb:hover{background:gray}#input-email{position:relative}#input-email #suggestions{position:absolute;max-height:400px;width:100%;z-index:100;margin-top:.25rem;padding:.25rem;overflow:hidden;font:var(--tui-font-text-m);border:1px solid var(--tui-base-04);border-radius:var(--tui-radius-m);background-color:var(--tui-base-01);box-shadow:0 1.5rem 1rem #00000008,0 .75rem .75rem #0000000a,0 .25rem .375rem #0000000d;animation-name:slide;animation-duration:.5s}#input-email #suggestions #suggestions-container{overflow:auto}#input-email #suggestions #suggestions-container .suggestion{min-height:2.75rem;padding:.5rem .625rem;display:flex;align-items:center;cursor:pointer}#input-email #suggestions #suggestions-container .suggestion.hover{background-color:#f2f2f2;border-radius:var(--tui-radius-s)}#input-email #suggestions.input-email-l{margin-top:.25rem!important;padding:.25rem!important;font:var(--tui-font-text-m)!important;border-radius:var(--tui-radius-m)!important}#input-email #suggestions.input-email-l #suggestions-container .suggestion{min-height:2.75rem!important;padding:.5rem .625rem!important}#input-email #suggestions.input-email-l #suggestions-container .suggestion.hover{border-radius:var(--tui-radius-s)!important}#input-email #suggestions.input-email-m{margin-top:.25rem!important;padding:.25rem!important;font:var(--tui-font-text-s)!important;border-radius:var(--tui-radius-m)!important}#input-email #suggestions.input-email-m #suggestions-container .suggestion{min-height:2.75rem!important;padding:.375rem .5rem!important}#input-email #suggestions.input-email-m #suggestions-container .suggestion.hover{border-radius:var(--tui-radius-s)!important}#input-email #suggestions.input-email-s{margin-top:.25rem!important;padding:.25rem!important;font:var(--tui-font-text-s)!important;border-radius:var(--tui-radius-m)!important}#input-email #suggestions.input-email-s #suggestions-container .suggestion{min-height:2rem!important;padding:.3125rem .5rem!important}#input-email #suggestions.input-email-s #suggestions-container .suggestion.hover{border-radius:var(--tui-radius-s)!important}@keyframes slide{0%{opacity:0}to{opacity:1}}\n"], components: [{ type: i1.TuiInputComponent, selector: "tui-input" }, { type: i2.TuiTextfieldComponent, selector: "input[tuiTextfield], textarea[tuiTextfield]" }], directives: [{ type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i1.TuiInputDirective, selector: "tui-input" }, { type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i3.FormControlName, selector: "[formControlName]", inputs: ["disabled", "formControlName", "ngModel"], outputs: ["ngModelChange"] }, { type: i2.TuiHintOptionsDirective, selector: "[tuiHintContent]", inputs: ["tuiHintContent", "tuiHintDirection", "tuiHintAppearance", "tuiHintShowDelay", "tuiHintHideDelay"] }, { type: i2.TuiTextfieldCleanerDirective, selector: "[tuiTextfieldCleaner]", inputs: ["tuiTextfieldCleaner"] }, { type: i2.TuiTextfieldIconLeftDirective, selector: "[tuiTextfieldIconLeft]", inputs: ["tuiTextfieldIconLeft"] }, { type: i2.TuiTextfieldPostfixDirective, selector: "[tuiTextfieldPostfix]", inputs: ["tuiTextfieldPostfix"] }, { type: i2.TuiTextfieldPrefixDirective, selector: "[tuiTextfieldPrefix]", inputs: ["tuiTextfieldPrefix"] }, { type: i2.TuiTextfieldSizeDirective, selector: "[tuiTextfieldSize]", inputs: ["tuiTextfieldSize"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: KwikUIInputEmailComponent, decorators: [{ type: Component, args: [{ selector: "kwikui-input-email", templateUrl: "./input-email.component.html", styleUrls: ["./input-email.component.scss"], providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef((() => KwikUIInputEmailComponent)), multi: true } ] }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { case: [{ type: Input }], disabled: [{ type: Input }], domains: [{ type: Input }], focus: [{ type: Input }], formControl: [{ type: Input }], formControlName: [{ type: Input }], hintContent: [{ type: Input }], icon: [{ type: Input }], id: [{ type: Input }], invalid: [{ type: Input }], label: [{ type: Input }], placeholder: [{ type: Input }], postfix: [{ type: Input }], prefix: [{ type: Input }], properties: [{ type: Input }], size: [{ type: Input }], validators: [{ type: Input }], getKeyValue: [{ type: Output }], inputEmail: [{ type: ViewChild, args: ["inputEmail"] }], dropdown: [{ type: ViewChild, args: ["dropdown"] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-email.component.js","sourceRoot":"","sources":["../../../../../../../projects/kwikui/src/lib/components/custom/input-email/input-email.component.ts","../../../../../../../projects/kwikui/src/lib/components/custom/input-email/input-email.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewChild\n} from \"@angular/core\";\nimport {\n  ControlValueAccessor,\n  FormControl,\n  FormGroup,\n  NG_VALUE_ACCESSOR\n} from \"@angular/forms\";\nimport { Subject, Subscription } from \"rxjs\";\nimport { debounceTime, distinctUntilChanged } from \"rxjs/operators\";\nimport { throwErrorMessage } from \"../../../helpers/kwikui.common.helpers\";\nimport { DEFAULT_VALUES, VALIDATE_KEY_VALUES } from \"./input-email.constants\";\nimport {\n  TKwikUIInputEmailCase,\n  TKwikUIInputEmailSize\n} from \"./input-email.definitions\";\nimport { isValidKeyValue } from \"./input-email.validation\";\n\n@Component({\n  selector: \"kwikui-input-email\",\n  templateUrl: \"./input-email.component.html\",\n  styleUrls: [\"./input-email.component.scss\"],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => KwikUIInputEmailComponent),\n      multi: true\n    }\n  ]\n})\nexport class KwikUIInputEmailComponent\n  implements OnInit, OnChanges, ControlValueAccessor {\n  @Input() case: TKwikUIInputEmailCase = DEFAULT_VALUES.case;\n\n  @Input() disabled: boolean = DEFAULT_VALUES.disabled;\n\n  @Input() domains: string[] = DEFAULT_VALUES.domains;\n\n  @Input() focus: boolean = DEFAULT_VALUES.focus;\n\n  @Input() formControl: FormControl = new FormControl({});\n\n  @Input() formControlName: string = DEFAULT_VALUES.formControlName;\n\n  @Input() hintContent: string = DEFAULT_VALUES.hintContent;\n\n  @Input() icon: string = DEFAULT_VALUES.icon;\n\n  @Input() id: string = DEFAULT_VALUES.id;\n\n  @Input() invalid: boolean = DEFAULT_VALUES.invalid;\n\n  @Input() label: string = DEFAULT_VALUES.label;\n\n  @Input() placeholder: string = DEFAULT_VALUES.placeholder;\n\n  @Input() postfix: string = DEFAULT_VALUES.postfix;\n\n  @Input() prefix: string = DEFAULT_VALUES.prefix;\n\n  @Input() properties = { readOnly: false };\n\n  @Input() size: TKwikUIInputEmailSize = DEFAULT_VALUES.size;\n\n  @Input() validators = { required: true };\n\n  @Output() getKeyValue: EventEmitter<any> = new EventEmitter<any>();\n\n  @ViewChild(\"inputEmail\") inputEmail: ElementRef;\n\n  @ViewChild(\"dropdown\") dropdown: ElementRef;\n\n  dropdownPosition: string = undefined;\n\n  suggestions: string[] = [];\n\n  domainSelected: string = undefined;\n\n  formGroup: FormGroup = new FormGroup({});\n\n  subscriptions = new Subscription();\n\n  private inputSubject = new Subject<string>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.formGroup.addControl(this.formControlName, this.formControl);\n    this.focus = this.focus === undefined ? false : this.focus;\n    this.invalid =\n      this.invalid === undefined ? false : this.invalid && this.focus;\n    this.validators = { ...this.validators };\n    this.properties = { ...this.properties };\n    this.changeCase();\n    this.setDisabled();\n\n    this.subscriptions.add(\n      this.formGroup.controls[this.formControlName].valueChanges\n        .pipe(distinctUntilChanged()) // makes sure the value has actually changed.\n        .subscribe((value) => this.handleInputValueChange(value))\n    );\n\n    this.inputSubject.pipe(debounceTime(300)).subscribe(() => {\n      this.updateSuggestions();\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const verifyChange = (key: string) => {\n      return changes.hasOwnProperty(key) && !changes[key].firstChange;\n    };\n\n    for (const change of Object.entries(changes)) {\n      const key: string = change[0];\n      const value: any = change[1].currentValue;\n      this.validateInputProperty(key, value);\n    }\n\n    if (verifyChange(\"formControl\")) {\n      this.formControl = changes.formControl.currentValue;\n    }\n    if (verifyChange(\"invalid\")) {\n      this.invalid = Boolean(changes.invalid.currentValue);\n      this.setError();\n    }\n    if (verifyChange(\"focus\")) {\n      this.focus = Boolean(changes.focus.currentValue);\n    }\n    if (verifyChange(\"disabled\")) {\n      this.disabled = Boolean(changes.disabled.currentValue);\n      this.setDisabled();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    // ensure when component is destroyed the subscription is also and not left open.\n  }\n\n  private validateInputProperty(key: string, value: any): void {\n    if (VALIDATE_KEY_VALUES[key] && !isValidKeyValue(key, value)) {\n      this[key] = DEFAULT_VALUES[key];\n      throwErrorMessage(\n        \"kwikui-input-email\",\n        this.id,\n        key,\n        value,\n        DEFAULT_VALUES[key]\n      );\n    }\n  }\n\n  /**\n   * @description Handles setting up of error and focus on the input field is it is invalid\n   */\n  setDisabled() {\n    if (this.formGroup.controls[this.formControlName] !== undefined) {\n      if (this.disabled === true) {\n        this.formGroup.controls[this.formControlName].disable({\n          emitEvent: false\n        });\n      } else {\n        this.formGroup.controls[this.formControlName].enable({\n          emitEvent: false\n        });\n      }\n    }\n  }\n\n  /**\n   * @description Handles setting up of error and focus on the input field is it is invalid\n   */\n  setError() {\n    if (\n      this.formGroup.controls[this.formControlName] !== undefined &&\n      this.formGroup.controls[this.formControlName].invalid\n    ) {\n      this.invalid = true;\n      this.focus = true;\n    } else {\n      this.invalid = false;\n      this.focus = false;\n    }\n  }\n\n  /**\n   * @description Handle changing of case based on 'case' props\n   */\n  changeCase() {\n    switch (this.case) {\n      case \"upper\":\n        this.formGroup.patchValue({\n          [this.formControlName]:\n            this.formGroup.controls[\n              this.formControlName\n            ].value.toLocaleUpperCase()\n        });\n        break;\n      case \"lower\":\n        this.formGroup.patchValue({\n          [this.formControlName]:\n            this.formGroup.controls[\n              this.formControlName\n            ].value.toLocaleLowerCase()\n        });\n        break;\n      default:\n        break;\n    }\n  }\n\n  /**\n   * @description Handles firing of 2 events on (keyup) event\n   *\n   * @param value\n   */\n  handleInputValueChange(value: any) {\n    this.changeCase();\n    this.inputSubject.next(this.formControl.value);\n    if (value === this.formControl.value) {\n      this.emitEvent(this.getKeyValue, {\n        key: this.formControlName,\n        value: this.formControl.value\n      });\n    }\n  }\n\n  /**\n   * @description Update the suggestion domains\n   */\n  updateSuggestions() {\n    const atIndex = this.formControl.value.lastIndexOf(\"@\");\n    this.suggestions = [];\n\n    if (!this.domainSelected && atIndex != -1) {\n      const domainInput = this.formControl.value.substring(atIndex + 1);\n      this.suggestions = this.domains.filter((domain) =>\n        domain.startsWith(domainInput)\n      );\n      this.cdr.detectChanges();\n      this.checkDropdownPosition();\n    }\n\n    this.domainSelected = undefined;\n  }\n\n  /**\n   * @description Handles Position of Suggestion Dropdown\n   */\n  checkDropdownPosition() {\n    if (this.dropdown) {\n      const rectRelative =\n        this.inputEmail.nativeElement.getBoundingClientRect();\n      const rectAbsolute = this.dropdown.nativeElement.getBoundingClientRect();\n\n      if (rectAbsolute.bottom + rectRelative.height > window.innerHeight) {\n        this.dropdownPosition = `-${rectAbsolute.height + 8}px`;\n      } else {\n        this.dropdownPosition = \"100%\";\n      }\n    }\n  }\n\n  /**\n   * @description Handle selection of domains\n   */\n  selectDomain(domain: string) {\n    this.formControl.setValue(\n      this.formControl.value.replace(/@[\\w.-]*$/, `@${domain}`)\n    );\n    this.suggestions = [];\n    this.domainSelected = domain;\n  }\n\n  handleFocusedChange(e: any) {\n    if (!this.invalid) {\n      this.focus = e;\n    }\n  }\n\n  emitEvent(event: any, data: any) {\n    event.emit(data);\n  }\n\n  /** Method Implementations for Abstract Control */\n  writeValue(value: any): void {}\n\n  registerOnChange(fn: any): void {}\n\n  registerOnTouched(fn: any): void {}\n\n  setDisabledState?(isDisabled: boolean): void {}\n}\n","<div\n  id=\"input-email\"\n  [formGroup]=\"formGroup\"\n  #inputEmail\n>\n  <tui-input\n    [formControlName]=\"formControlName\"\n    [nativeId]=\"id\"\n    [pseudoFocus]=\"focus ?? null\"\n    [pseudoInvalid]=\"invalid === undefined ? null : invalid\"\n    [readOnly]=\"properties?.readOnly ?? false\"\n    [tuiHintContent]=\"hintContent\"\n    [tuiTextfieldCleaner]=\"true\"\n    [tuiTextfieldIconLeft]=\"icon\"\n    [tuiTextfieldPostfix]=\"postfix\"\n    [tuiTextfieldPrefix]=\"prefix\"\n    [tuiTextfieldSize]=\"size\"\n    (focusedChange)=\"handleFocusedChange($event)\"\n  >\n    {{ label }}\n    <span\n      class=\"tui-required\"\n      *ngIf=\"validators.required\"\n    ></span>\n    <input\n      tuiTextfield\n      [attr.type]=\"'email'\"\n      [attr.placeholder]=\"placeholder ?? ''\"\n      [attr.required]=\"validators.required\"\n    />\n  </tui-input>\n  <div\n    #dropdown\n    [class.input-email-s]=\"size === 's'\"\n    [class.input-email-m]=\"size === 'm'\"\n    [class.input-email-l]=\"size === 'l'\"\n    [style.top]=\"dropdownPosition\"\n    id=\"suggestions\"\n    *ngIf=\"suggestions && suggestions.length > 0\"\n  >\n    <div id=\"suggestions-container\"\n      ><div\n        class=\"suggestion\"\n        *ngFor=\"let domain of suggestions\"\n        (click)=\"selectDomain(domain)\"\n        >{{ domain }}</div\n      ></div\n    >\n  </div>\n</div>\n"]}