UNPKG

ngx-tiptap-editor

Version:

[![Build and Publish](https://github.com/HuiiBuh/ngx-tiptap-editor/actions/workflows/publish.yml/badge.svg)](https://github.com/HuiiBuh/ngx-tiptap-editor/actions/workflows/publish.yml) [![Deploy to Github Pages](https://github.com/HuiiBuh/ngx-tiptap-edito

202 lines 37.8 kB
import { DOCUMENT } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Inject, Input, Output, SecurityContext, ViewChild } from '@angular/core'; import { fromEvent, merge, Subject } from 'rxjs'; import { filter, startWith, switchMap, takeUntil } from 'rxjs/operators'; import { FadeInAnimation } from '../../animations'; import { ExpandHeight } from './select.animations'; import * as i0 from "@angular/core"; import * as i1 from "@angular/platform-browser"; import * as i2 from "@angular/common"; export class OptionComponent { constructor(element) { this.element = element; this.onSelect = new EventEmitter(); this.enforceHeight = false; this.useHtml = false; this._disabled = false; } set disabled(value) { if (this.option && value) { this.option.nativeElement.setAttribute('disabled', 'true'); } else { this.option && this.option.nativeElement.removeAttribute('disabled'); } this._disabled = false; } setSelected(value) { this.addOrRemoveClass(value, 'active'); } emit($event) { $event.preventDefault(); this.onSelect.emit(this); } getContent() { return this.useHtml ? this.element.nativeElement.innerHTML : this.element.nativeElement.textContent; } addOrRemoveClass(add, className) { const operation = add ? 'add' : 'remove'; this.option && this.option.nativeElement.classList[operation](className); } } OptionComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: OptionComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); OptionComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.1.1", type: OptionComponent, selector: "tip-option[value]", inputs: { value: "value", enforceHeight: "enforceHeight", useHtml: "useHtml", disabled: "disabled" }, viewQueries: [{ propertyName: "option", first: true, predicate: ["option"], descendants: true }], ngImport: i0, template: ` <button type="button" (click)="emit($event)" (keydown.enter)="emit($event)" class="select-option select-overflow-wrapper" #option> <ng-content></ng-content> </button>`, isInline: true, styles: [":host-context(tip-select){display:flex}.select-wrapper{position:relative;display:inline-flex;align-items:center;-webkit-user-select:none;user-select:none;outline-offset:-1}.select-preview{display:inline-flex;overflow:hidden;align-items:center;box-sizing:border-box;width:100%;height:var(--tip-select-preview-height);padding:0 .2rem;cursor:pointer;border:solid 1px var(--tip-border-color);border-radius:calc(var(--tip-border-radius) / 2)}.select-preview:focus{color:var(--tip-active-color)}.select-preview.icon-placeholder{padding-right:var(--tip-select-preview-height)}.select-overflow-wrapper{overflow:hidden;box-sizing:border-box;width:100%;white-space:nowrap;display:flex;align-items:center;text-overflow:ellipsis}.select-options-overlay{position:absolute;box-sizing:border-box;z-index:1;top:0;width:100%;margin-top:4px;border-radius:calc(var(--tip-border-radius) / 2);border:solid 1px var(--tip-border-color);cursor:pointer;transform:translateY(var(--tip-select-preview-height));background-color:var(--tip-background-color);overflow-y:hidden}.select-icon{position:absolute;top:50%;right:0;height:var(--tip-select-preview-height);transform:translateY(-50%);transition:.3s rotate;fill:var(--tip-text-color)}.select-icon.rotate180{transform:rotate(180deg) translateY(50%)}::ng-deep tip-select .select-preview-content{height:var(--tip-select-preview-height)}::ng-deep tip-select .select-preview-content>.select-option{height:var(--tip-select-preview-height)}::ng-deep tip-option:last-child .select-option{border-bottom:none}.select-option{padding:.2rem;transition:color .3s;border-bottom:solid 1px var(--tip-border-color);border-left:none;border-top:none;border-right:none;background-color:transparent;color:var(--tip-text-color)}.select-option[disabled]{color:var(--tip-disabled-color)}.select-option:not([disabled]){cursor:pointer}.select-option:not([disabled]):hover,.select-option:not([disabled]).tip-active,.select-option:not([disabled]):focus{color:var(--tip-active-color)}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: OptionComponent, decorators: [{ type: Component, args: [{ selector: 'tip-option[value]', template: ` <button type="button" (click)="emit($event)" (keydown.enter)="emit($event)" class="select-option select-overflow-wrapper" #option> <ng-content></ng-content> </button>`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [":host-context(tip-select){display:flex}.select-wrapper{position:relative;display:inline-flex;align-items:center;-webkit-user-select:none;user-select:none;outline-offset:-1}.select-preview{display:inline-flex;overflow:hidden;align-items:center;box-sizing:border-box;width:100%;height:var(--tip-select-preview-height);padding:0 .2rem;cursor:pointer;border:solid 1px var(--tip-border-color);border-radius:calc(var(--tip-border-radius) / 2)}.select-preview:focus{color:var(--tip-active-color)}.select-preview.icon-placeholder{padding-right:var(--tip-select-preview-height)}.select-overflow-wrapper{overflow:hidden;box-sizing:border-box;width:100%;white-space:nowrap;display:flex;align-items:center;text-overflow:ellipsis}.select-options-overlay{position:absolute;box-sizing:border-box;z-index:1;top:0;width:100%;margin-top:4px;border-radius:calc(var(--tip-border-radius) / 2);border:solid 1px var(--tip-border-color);cursor:pointer;transform:translateY(var(--tip-select-preview-height));background-color:var(--tip-background-color);overflow-y:hidden}.select-icon{position:absolute;top:50%;right:0;height:var(--tip-select-preview-height);transform:translateY(-50%);transition:.3s rotate;fill:var(--tip-text-color)}.select-icon.rotate180{transform:rotate(180deg) translateY(50%)}::ng-deep tip-select .select-preview-content{height:var(--tip-select-preview-height)}::ng-deep tip-select .select-preview-content>.select-option{height:var(--tip-select-preview-height)}::ng-deep tip-option:last-child .select-option{border-bottom:none}.select-option{padding:.2rem;transition:color .3s;border-bottom:solid 1px var(--tip-border-color);border-left:none;border-top:none;border-right:none;background-color:transparent;color:var(--tip-text-color)}.select-option[disabled]{color:var(--tip-disabled-color)}.select-option:not([disabled]){cursor:pointer}.select-option:not([disabled]):hover,.select-option:not([disabled]).tip-active,.select-option:not([disabled]):focus{color:var(--tip-active-color)}\n"] }] }], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { value: [{ type: Input }], enforceHeight: [{ type: Input }], useHtml: [{ type: Input }], option: [{ type: ViewChild, args: ['option'] }], disabled: [{ type: Input }] } }); // @dynamic export class SelectComponent { constructor(cd, element, ngZone, renderer2, sanitizer, document) { this.cd = cd; this.element = element; this.ngZone = ngZone; this.renderer2 = renderer2; this.sanitizer = sanitizer; this.document = document; this.width = '150px'; this.placeholder = ''; this.defaultValue = ''; this.showIcon = true; this.disablePreviewSanitation = false; // tslint:disable-next-line:no-output-native this.change = new EventEmitter(); // Is the dropdown visible this.visible = false; this.destroy$ = new Subject(); } get value() { return this._value; } set value(value) { // Nothing changed if (value === this._value) return; this._value = value; this.updateComponent(false); } ngOnInit() { this.ngZone.runOutsideAngular(() => { fromEvent(this.document, 'keyup').pipe(filter(e => e.key === 'Escape'), filter(() => this.visible), takeUntil(this.destroy$)).subscribe(() => { this.ngZone.run(() => { this.visible = false; this.cd.markForCheck(); }); }); fromEvent(this.document, 'click').pipe(filter(e => !this.element.nativeElement.contains(e.target) && this.visible), filter(e => ( // Dont trigger close if the event comes from the own toggle button, or its children e.target === this.toggleElement?.nativeElement || !!this.selectPreview && !this.selectPreview.nativeElement.contains(e.target))), takeUntil(this.destroy$)).subscribe(() => { this.ngZone.run(() => { this.visible = false; this.cd.markForCheck(); }); }); }); } ngAfterViewInit() { // Subscribe to click events on the options const options = this.optionList; this.optionList.changes.pipe(startWith(...options), switchMap(() => merge(...options.map(o => o.onSelect))), takeUntil(this.destroy$)).subscribe(component => { // Nothing changed if (this._value === component.value) return; this._value = component.value; this.visible = false; this.updateComponent(true); }); // Update the selected value for the initial select this.updateComponent(false); } ngOnDestroy() { this.destroy$.next(); this.destroy$.complete(); } toggle() { this.visible = !this.visible; } /** * Select the current component depending on the selected value * @param emitUpdate Should the change be propagated */ updateComponent(emitUpdate) { // No options => no update if (!this.optionList) return; // If no value is provided use the default value if (!this.value && this.defaultValue) { this._value = this.defaultValue; } // Deselect all this.optionList.forEach(o => o.setSelected(false)); // Select right one const selectedComponent = this.optionList.find(o => o.value === this._value); let previewText = this.placeholder; // A component is selected, so select the component if (selectedComponent) { selectedComponent.setSelected(true); previewText = selectedComponent.getContent(); emitUpdate && this.change.emit(selectedComponent.value); } let sanitizedHtml = previewText; if (!this.disablePreviewSanitation) { sanitizedHtml = this.sanitizer.sanitize(SecurityContext.HTML, previewText); } // Dont update if ou don't have to updaet if (this.selectPreview?.nativeElement.innerHTML === sanitizedHtml) return; this.selectPreview && this.renderer2.setProperty(this.selectPreview?.nativeElement, 'innerHTML', sanitizedHtml); } } SelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: SelectComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.ElementRef }, { token: i0.NgZone }, { token: i0.Renderer2 }, { token: i1.DomSanitizer }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Component }); SelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.1.1", type: SelectComponent, selector: "tip-select", inputs: { width: "width", placeholder: "placeholder", defaultValue: "defaultValue", showIcon: "showIcon", disablePreviewSanitation: "disablePreviewSanitation", value: "value" }, outputs: { change: "change" }, queries: [{ propertyName: "optionList", predicate: OptionComponent, descendants: true }], viewQueries: [{ propertyName: "selectPreview", first: true, predicate: ["selectPreview"], descendants: true }, { propertyName: "toggleElement", first: true, predicate: ["toggleElement"], descendants: true }], ngImport: i0, template: "<div [ngStyle]=\"{width: width}\" class=\"select-wrapper\">\n <div (click)=\"toggle()\" (keydown.enter)=\"toggle()\" [class.icon-placeholder]=\"showIcon\"\n class=\"select-preview\"\n tabindex=\"0\" #toggleElement>\n <div #selectPreview\n class=\"select-overflow-wrapper select-preview-content\">\n </div>\n <i *ngIf=\"showIcon\" [class.rotate180]=\"visible\" class=\"select-icon\">\n <svg focusable=\"false\" height=\"24px\" viewBox=\"0 0 24 24\" width=\"24px\">\n <path d=\"M7 10l5 5 5-5z\"/>\n </svg>\n </i>\n </div>\n <div *ngIf=\"visible\" class=\"select-options-overlay\" @expandHeight>\n <ng-content></ng-content>\n </div>\n</div>\n\n", styles: [":host-context(tip-select){display:flex}.select-wrapper{position:relative;display:inline-flex;align-items:center;-webkit-user-select:none;user-select:none;outline-offset:-1}.select-preview{display:inline-flex;overflow:hidden;align-items:center;box-sizing:border-box;width:100%;height:var(--tip-select-preview-height);padding:0 .2rem;cursor:pointer;border:solid 1px var(--tip-border-color);border-radius:calc(var(--tip-border-radius) / 2)}.select-preview:focus{color:var(--tip-active-color)}.select-preview.icon-placeholder{padding-right:var(--tip-select-preview-height)}.select-overflow-wrapper{overflow:hidden;box-sizing:border-box;width:100%;white-space:nowrap;display:flex;align-items:center;text-overflow:ellipsis}.select-options-overlay{position:absolute;box-sizing:border-box;z-index:1;top:0;width:100%;margin-top:4px;border-radius:calc(var(--tip-border-radius) / 2);border:solid 1px var(--tip-border-color);cursor:pointer;transform:translateY(var(--tip-select-preview-height));background-color:var(--tip-background-color);overflow-y:hidden}.select-icon{position:absolute;top:50%;right:0;height:var(--tip-select-preview-height);transform:translateY(-50%);transition:.3s rotate;fill:var(--tip-text-color)}.select-icon.rotate180{transform:rotate(180deg) translateY(50%)}::ng-deep tip-select .select-preview-content{height:var(--tip-select-preview-height)}::ng-deep tip-select .select-preview-content>.select-option{height:var(--tip-select-preview-height)}::ng-deep tip-option:last-child .select-option{border-bottom:none}.select-option{padding:.2rem;transition:color .3s;border-bottom:solid 1px var(--tip-border-color);border-left:none;border-top:none;border-right:none;background-color:transparent;color:var(--tip-text-color)}.select-option[disabled]{color:var(--tip-disabled-color)}.select-option:not([disabled]){cursor:pointer}.select-option:not([disabled]):hover,.select-option:not([disabled]).tip-active,.select-option:not([disabled]):focus{color:var(--tip-active-color)}\n"], directives: [{ type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], animations: [FadeInAnimation, ExpandHeight], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: SelectComponent, decorators: [{ type: Component, args: [{ selector: 'tip-select', animations: [FadeInAnimation, ExpandHeight], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div [ngStyle]=\"{width: width}\" class=\"select-wrapper\">\n <div (click)=\"toggle()\" (keydown.enter)=\"toggle()\" [class.icon-placeholder]=\"showIcon\"\n class=\"select-preview\"\n tabindex=\"0\" #toggleElement>\n <div #selectPreview\n class=\"select-overflow-wrapper select-preview-content\">\n </div>\n <i *ngIf=\"showIcon\" [class.rotate180]=\"visible\" class=\"select-icon\">\n <svg focusable=\"false\" height=\"24px\" viewBox=\"0 0 24 24\" width=\"24px\">\n <path d=\"M7 10l5 5 5-5z\"/>\n </svg>\n </i>\n </div>\n <div *ngIf=\"visible\" class=\"select-options-overlay\" @expandHeight>\n <ng-content></ng-content>\n </div>\n</div>\n\n", styles: [":host-context(tip-select){display:flex}.select-wrapper{position:relative;display:inline-flex;align-items:center;-webkit-user-select:none;user-select:none;outline-offset:-1}.select-preview{display:inline-flex;overflow:hidden;align-items:center;box-sizing:border-box;width:100%;height:var(--tip-select-preview-height);padding:0 .2rem;cursor:pointer;border:solid 1px var(--tip-border-color);border-radius:calc(var(--tip-border-radius) / 2)}.select-preview:focus{color:var(--tip-active-color)}.select-preview.icon-placeholder{padding-right:var(--tip-select-preview-height)}.select-overflow-wrapper{overflow:hidden;box-sizing:border-box;width:100%;white-space:nowrap;display:flex;align-items:center;text-overflow:ellipsis}.select-options-overlay{position:absolute;box-sizing:border-box;z-index:1;top:0;width:100%;margin-top:4px;border-radius:calc(var(--tip-border-radius) / 2);border:solid 1px var(--tip-border-color);cursor:pointer;transform:translateY(var(--tip-select-preview-height));background-color:var(--tip-background-color);overflow-y:hidden}.select-icon{position:absolute;top:50%;right:0;height:var(--tip-select-preview-height);transform:translateY(-50%);transition:.3s rotate;fill:var(--tip-text-color)}.select-icon.rotate180{transform:rotate(180deg) translateY(50%)}::ng-deep tip-select .select-preview-content{height:var(--tip-select-preview-height)}::ng-deep tip-select .select-preview-content>.select-option{height:var(--tip-select-preview-height)}::ng-deep tip-option:last-child .select-option{border-bottom:none}.select-option{padding:.2rem;transition:color .3s;border-bottom:solid 1px var(--tip-border-color);border-left:none;border-top:none;border-right:none;background-color:transparent;color:var(--tip-text-color)}.select-option[disabled]{color:var(--tip-disabled-color)}.select-option:not([disabled]){cursor:pointer}.select-option:not([disabled]):hover,.select-option:not([disabled]).tip-active,.select-option:not([disabled]):focus{color:var(--tip-active-color)}\n"] }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }, { type: i0.NgZone }, { type: i0.Renderer2 }, { type: i1.DomSanitizer }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }]; }, propDecorators: { width: [{ type: Input }], placeholder: [{ type: Input }], defaultValue: [{ type: Input }], showIcon: [{ type: Input }], disablePreviewSanitation: [{ type: Input }], change: [{ type: Output }], optionList: [{ type: ContentChildren, args: [OptionComponent, { descendants: true }] }], selectPreview: [{ type: ViewChild, args: ['selectPreview'] }], toggleElement: [{ type: ViewChild, args: ['toggleElement'] }], value: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-tiptap-editor/src/lib/components/select/select.component.ts","../../../../../../projects/ngx-tiptap-editor/src/lib/components/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n  QueryList,\n  Renderer2,\n  SecurityContext,\n  ViewChild\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { fromEvent, merge, Subject } from 'rxjs';\nimport { filter, startWith, switchMap, takeUntil } from 'rxjs/operators';\nimport { FadeInAnimation } from '../../animations';\nimport { ExpandHeight } from './select.animations';\n\n@Component({\n  selector: 'tip-option[value]',\n  template: `\n    <button type=\"button\" (click)=\"emit($event)\" (keydown.enter)=\"emit($event)\"\n            class=\"select-option select-overflow-wrapper\" #option>\n      <ng-content></ng-content>\n    </button>`,\n  styleUrls: ['select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class OptionComponent {\n  public onSelect = new EventEmitter<OptionComponent>();\n  @Input() value: any;\n  @Input() enforceHeight = false;\n  @Input() useHtml = false;\n  public _disabled = false;\n  @ViewChild('option') private option: ElementRef<HTMLDivElement> | undefined;\n\n  constructor(private element: ElementRef) {\n  }\n\n  @Input() set disabled(value: boolean) {\n    if (this.option && value) {\n      this.option.nativeElement.setAttribute('disabled', 'true');\n    } else {\n      this.option && this.option.nativeElement.removeAttribute('disabled');\n    }\n    this._disabled = false;\n  }\n\n  public setSelected(value: boolean): void {\n    this.addOrRemoveClass(value, 'active');\n  }\n\n  public emit($event: MouseEvent | Event): void {\n    $event.preventDefault();\n    this.onSelect.emit(this);\n  }\n\n  public getContent(): string {\n    return this.useHtml ? this.element.nativeElement.innerHTML : this.element.nativeElement.textContent;\n  }\n\n  private addOrRemoveClass(add: boolean, className: string): void {\n    const operation = add ? 'add' : 'remove';\n    this.option && this.option.nativeElement.classList[operation](className);\n  }\n\n}\n\n// @dynamic\n@Component({\n  selector: 'tip-select',\n  templateUrl: './select.component.html',\n  animations: [FadeInAnimation, ExpandHeight],\n  styleUrls: ['./select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class SelectComponent implements AfterViewInit, OnDestroy, OnInit {\n\n  @Input() public width = '150px';\n  @Input() public placeholder = '';\n  @Input() public defaultValue = '';\n  @Input() public showIcon = true;\n  @Input() public disablePreviewSanitation = false;\n  // tslint:disable-next-line:no-output-native\n  @Output() public change = new EventEmitter<any>();\n  // Is the dropdown visible\n  public visible = false;\n  @ContentChildren(OptionComponent, {descendants: true}) private optionList!: QueryList<OptionComponent>;\n  @ViewChild('selectPreview') private selectPreview: ElementRef<HTMLDivElement> | undefined;\n  @ViewChild('toggleElement') private toggleElement: ElementRef<HTMLDivElement> | undefined;\n  private destroy$ = new Subject();\n  private _value: any;\n\n  constructor(\n    private cd: ChangeDetectorRef,\n    private element: ElementRef,\n    private ngZone: NgZone,\n    private renderer2: Renderer2,\n    private sanitizer: DomSanitizer,\n    @Inject(DOCUMENT) private document: Document\n  ) {\n  }\n\n  get value(): any {\n    return this._value;\n  }\n\n  @Input()\n  set value(value: any) {\n    // Nothing changed\n    if (value === this._value) return;\n\n    this._value = value;\n    this.updateComponent(false);\n  }\n\n  public ngOnInit(): void {\n    this.ngZone.runOutsideAngular(() => {\n      fromEvent<KeyboardEvent>(this.document, 'keyup').pipe(\n        filter(e => e.key === 'Escape'),\n        filter(() => this.visible),\n        takeUntil(this.destroy$)\n      ).subscribe(() => {\n        this.ngZone.run(() => {\n          this.visible = false;\n          this.cd.markForCheck();\n        });\n      });\n      fromEvent<KeyboardEvent>(this.document, 'click').pipe(\n        filter(e => !this.element.nativeElement.contains(e.target as Node) && this.visible),\n        filter(e => (\n          // Dont trigger close if the event comes from the own toggle button, or its children\n          (e.target as HTMLElement) === this.toggleElement?.nativeElement ||\n          !!this.selectPreview && !this.selectPreview.nativeElement.contains((e.target as HTMLElement)))\n        ),\n        takeUntil(this.destroy$)\n      ).subscribe(() => {\n        this.ngZone.run(() => {\n          this.visible = false;\n          this.cd.markForCheck();\n        });\n      });\n    });\n  }\n\n  public ngAfterViewInit(): void {\n    // Subscribe to click events on the options\n    const options = this.optionList;\n    this.optionList.changes.pipe(\n      startWith(...options),\n      switchMap(() => merge(...options.map(o => o.onSelect))),\n      takeUntil(this.destroy$),\n    ).subscribe(component => {\n      // Nothing changed\n      if (this._value === component.value) return;\n      this._value = component.value;\n      this.visible = false;\n      this.updateComponent(true);\n    });\n    // Update the selected value for the initial select\n    this.updateComponent(false);\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  public toggle(): void {\n    this.visible = !this.visible;\n  }\n\n  /**\n   * Select the current component depending on the selected value\n   * @param emitUpdate Should the change be propagated\n   */\n  private updateComponent(emitUpdate: boolean): void {\n    // No options => no update\n    if (!this.optionList) return;\n\n    // If no value is provided use the default value\n    if (!this.value && this.defaultValue) {\n      this._value = this.defaultValue;\n    }\n\n    // Deselect all\n    this.optionList.forEach(o => o.setSelected(false));\n\n    // Select right one\n    const selectedComponent = this.optionList.find(o => o.value === this._value);\n\n    let previewText = this.placeholder;\n    // A component is selected, so select the component\n    if (selectedComponent) {\n      selectedComponent.setSelected(true);\n      previewText = selectedComponent.getContent();\n\n      emitUpdate && this.change.emit(selectedComponent.value);\n    }\n\n    let sanitizedHtml = previewText;\n    if (!this.disablePreviewSanitation) {\n      sanitizedHtml = this.sanitizer.sanitize(SecurityContext.HTML, previewText)!;\n    }\n\n    // Dont update if ou don't have to updaet\n    if (this.selectPreview?.nativeElement.innerHTML === sanitizedHtml) return;\n    this.selectPreview && this.renderer2.setProperty(this.selectPreview?.nativeElement, 'innerHTML', sanitizedHtml);\n  }\n}\n","<div [ngStyle]=\"{width: width}\" class=\"select-wrapper\">\n  <div (click)=\"toggle()\" (keydown.enter)=\"toggle()\" [class.icon-placeholder]=\"showIcon\"\n       class=\"select-preview\"\n       tabindex=\"0\" #toggleElement>\n    <div #selectPreview\n         class=\"select-overflow-wrapper select-preview-content\">\n    </div>\n    <i *ngIf=\"showIcon\" [class.rotate180]=\"visible\" class=\"select-icon\">\n      <svg focusable=\"false\" height=\"24px\" viewBox=\"0 0 24 24\" width=\"24px\">\n        <path d=\"M7 10l5 5 5-5z\"/>\n      </svg>\n    </i>\n  </div>\n  <div *ngIf=\"visible\" class=\"select-options-overlay\" @expandHeight>\n    <ng-content></ng-content>\n  </div>\n</div>\n\n"]}