UNPKG

carbon-components-angular

Version:
349 lines (345 loc) 32.4 kB
import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, HostListener, Input, Output, ViewChild } from "@angular/core"; import { AILabelPopoverDirective } from "./ai-label-popover.directive"; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "carbon-components-angular/button"; import * as i3 from "carbon-components-angular/icon"; import * as i4 from "./ai-label-popover.directive"; /** * AI-branded toggletip control (`cds-ai-label`). Renders an "AI" badge that opens a * popover; projected content and optional actions use `ng-content`. * * Get started with importing the module: * * ```typescript * import { AILabelModule } from 'carbon-components-angular'; * ``` * * ```html * <cds-ai-label size="md"> * <div> * <p>AI Explained</p> * <h2>84%</h2> * <p>Confidence score</p> * </div> * <div cdsAILabelActions> * <button cdsButton="ghost" size="sm">View details</button> * </div> * </cds-ai-label> * ``` * * `[cdsAILabelActions]` adds `cds--toggletip-actions` and `cds--ai-label-actions` * to its host. Place it as a **sibling** of the body content, both direct * children of `<cds-ai-label>`. `[cdsAILabelContent]` is an optional marker; the * `cds--ai-label-content` / `cds--toggletip-content` classes come from this * component’s template. * * [See demo](../../?path=/story/components-ai-label--default) */ export class AILabelComponent { constructor(elementRef) { this.elementRef = elementRef; this.aiLabelClass = true; /** * Show caret at the alignment position. */ this.caret = true; /** * Enable drop shadow around the popover container. */ this.dropShadow = false; /** * Enable high contrast for popover container. */ this.highContrast = true; /** * **Experimental**: Use floating-ui to position the tooltip. */ this.autoAlign = false; /** * Whether the callout is open. */ this.isOpen = false; /** * Emits when the callout is closed. */ this.onClose = new EventEmitter(); /** * Emits when the callout is opened. */ this.onOpen = new EventEmitter(); /** * Emits when `isOpen` changes (two-way binding). */ this.isOpenChange = new EventEmitter(); /** * Unique id used to associate the trigger button with the popover panel * via `aria-controls` / `id`. */ this.id = `ai-label-${AILabelComponent.labelCounter++}`; /** * Text inside the AI badge. */ this.aiText = "AI"; /** * Set badge shape: `"default"` (circular) or `"inline"` (pill, optional `textLabel`). */ this.kind = "default"; /** * Set badge size */ this.size = "xs"; /** * When `true`, shows the revert icon instead of the badge (AI-generated value * is active and can be reverted). */ this.revertActive = false; /** * Accessible label / tooltip for the revert icon button. */ this.revertLabel = "Revert to AI input"; /** * `aria-label` for the AI badge trigger (combined with `aiText` in `computedAriaLabel`). */ this.ariaLabel = "Show information"; /** * Emitted when the revert icon is clicked. */ this.revertClick = new EventEmitter(); this.documentClick = this.handleOutsideClick.bind(this); } get revertClass() { return this.revertActive; } /** * Horizontal shift along the alignment axis when `autoAlign` is on, matching * React `AILabel` (`alignmentAxisOffset={isSmallIcon ? -24 : 0}` on `Toggletip`). */ get alignmentAxisOffset() { return ["mini", "2xs", "xs"].includes(this.size) ? -24 : 0; } onPopoverIsOpenChange(open) { this.isOpen = open; this.isOpenChange.emit(open); } get triggerClasses() { return { "cds--toggletip-button": true, "cds--ai-label__button": true, [`cds--ai-label__button--${this.size}`]: true, [`cds--ai-label__button--${this.kind}`]: true, "cds--ai-label__button--inline-with-content": this.kind === "inline" && !!this.textLabel }; } /** * Trigger `aria-label`: `"${aiText} ${ariaLabel}"`, or * `"${aiText} ${textLabel}"` when `kind` is `"inline"` and `textLabel` is set. */ get computedAriaLabel() { const suffix = (this.kind === "inline" && this.textLabel) ? this.textLabel : this.ariaLabel; return `${this.aiText} ${suffix}`; } ngAfterViewInit() { if (this.isOpen) { document.addEventListener("click", this.documentClick); } } ngOnChanges(changes) { if (changes.revertActive && !changes.revertActive.firstChange && changes.revertActive.currentValue) { this.isOpen = false; document.removeEventListener("click", this.documentClick); } } ngOnDestroy() { document.removeEventListener("click", this.documentClick); } onTriggerClick(event) { const opening = !this.isOpen; if (opening) { document.addEventListener("click", this.documentClick); } else { document.removeEventListener("click", this.documentClick); } this.aiLabelPopover?.handleChange(opening, event); } onRevertButtonClick(event) { this.revertClick.emit(event); } hostkeys(event) { if (this.isOpen && event.key === "Escape") { event.stopPropagation(); document.removeEventListener("click", this.documentClick); this.aiLabelPopover?.handleChange(false, event); } } /** * Dismisses the popover when a click lands outside the host element. */ handleOutsideClick(event) { if (!this.elementRef.nativeElement.contains(event.target)) { this.aiLabelPopover?.handleChange(false, event); document.removeEventListener("click", this.documentClick); } } } AILabelComponent.labelCounter = 0; AILabelComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AILabelComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); AILabelComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: AILabelComponent, selector: "cds-ai-label, ibm-ai-label", inputs: { align: "align", caret: "caret", dropShadow: "dropShadow", highContrast: "highContrast", autoAlign: "autoAlign", isOpen: "isOpen", id: "id", aiText: "aiText", textLabel: "textLabel", kind: "kind", size: "size", revertActive: "revertActive", revertLabel: "revertLabel", ariaLabel: "ariaLabel" }, outputs: { onClose: "onClose", onOpen: "onOpen", isOpenChange: "isOpenChange", revertClick: "revertClick" }, host: { listeners: { "keyup": "hostkeys($event)" }, properties: { "class.cds--ai-label": "this.aiLabelClass", "class.cds--ai-label--revert": "this.revertClass" } }, viewQueries: [{ propertyName: "aiLabelPopover", first: true, predicate: ["aiLabelPopoverHost"], descendants: true, read: AILabelPopoverDirective }], usesOnChanges: true, ngImport: i0, template: ` <ng-container *ngIf="!revertActive"> <span #aiLabelPopoverHost cdsAILabelPopover class="cds--toggletip" [isOpen]="isOpen" (isOpenChange)="onPopoverIsOpenChange($event)" (onOpen)="onOpen.emit($event)" (onClose)="onClose.emit($event)" [align]="align" [caret]="caret" [dropShadow]="dropShadow" [highContrast]="highContrast" [autoAlign]="autoAlign" [alignmentAxisOffset]="alignmentAxisOffset"> <button type="button" [attr.aria-label]="computedAriaLabel" [attr.aria-expanded]="isOpen" [attr.aria-controls]="id" [ngClass]="triggerClasses" (click)="onTriggerClick($event)"> <span class="cds--ai-label__text">{{aiText}}</span> <span *ngIf="kind === 'inline' && textLabel" class="cds--ai-label__additional-text">{{textLabel}}</span> </button> <span [id]="id" class="cds--popover" aria-live="polite"> <span class="cds--popover-content cds--ai-label-content"> <div class="cds--toggletip-content"> <ng-content></ng-content> </div> <span *ngIf="autoAlign" class="cds--popover-caret cds--popover--auto-align"></span> </span> <span *ngIf="!autoAlign" class="cds--popover-caret"></span> </span> </span> </ng-container> <cds-icon-button *ngIf="revertActive" kind="ghost" size="sm" [description]="revertLabel" [autoAlign]="autoAlign" [buttonAttributes]="{ 'aria-label': revertLabel }" (click)="onRevertButtonClick($event)"> <svg cdsIcon="undo" size="16"></svg> </cds-icon-button> `, isInline: true, dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.IconButton, selector: "cds-icon-button, ibm-icon-button", inputs: ["buttonNgClass", "buttonAttributes", "buttonId", "kind", "size", "type", "isExpressive", "disabled", "description", "showTooltipWhenDisabled"], outputs: ["click", "focus", "blur", "tooltipClick"] }, { kind: "directive", type: i3.IconDirective, selector: "[cdsIcon], [ibmIcon]", inputs: ["ibmIcon", "cdsIcon", "size", "title", "ariaLabel", "ariaLabelledBy", "ariaHidden", "isFocusable"] }, { kind: "directive", type: i4.AILabelPopoverDirective, selector: "[cdsAILabelPopover]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AILabelComponent, decorators: [{ type: Component, args: [{ selector: "cds-ai-label, ibm-ai-label", changeDetection: ChangeDetectionStrategy.OnPush, template: ` <ng-container *ngIf="!revertActive"> <span #aiLabelPopoverHost cdsAILabelPopover class="cds--toggletip" [isOpen]="isOpen" (isOpenChange)="onPopoverIsOpenChange($event)" (onOpen)="onOpen.emit($event)" (onClose)="onClose.emit($event)" [align]="align" [caret]="caret" [dropShadow]="dropShadow" [highContrast]="highContrast" [autoAlign]="autoAlign" [alignmentAxisOffset]="alignmentAxisOffset"> <button type="button" [attr.aria-label]="computedAriaLabel" [attr.aria-expanded]="isOpen" [attr.aria-controls]="id" [ngClass]="triggerClasses" (click)="onTriggerClick($event)"> <span class="cds--ai-label__text">{{aiText}}</span> <span *ngIf="kind === 'inline' && textLabel" class="cds--ai-label__additional-text">{{textLabel}}</span> </button> <span [id]="id" class="cds--popover" aria-live="polite"> <span class="cds--popover-content cds--ai-label-content"> <div class="cds--toggletip-content"> <ng-content></ng-content> </div> <span *ngIf="autoAlign" class="cds--popover-caret cds--popover--auto-align"></span> </span> <span *ngIf="!autoAlign" class="cds--popover-caret"></span> </span> </span> </ng-container> <cds-icon-button *ngIf="revertActive" kind="ghost" size="sm" [description]="revertLabel" [autoAlign]="autoAlign" [buttonAttributes]="{ 'aria-label': revertLabel }" (click)="onRevertButtonClick($event)"> <svg cdsIcon="undo" size="16"></svg> </cds-icon-button> ` }] }], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { aiLabelClass: [{ type: HostBinding, args: ["class.cds--ai-label"] }], revertClass: [{ type: HostBinding, args: ["class.cds--ai-label--revert"] }], align: [{ type: Input }], caret: [{ type: Input }], dropShadow: [{ type: Input }], highContrast: [{ type: Input }], autoAlign: [{ type: Input }], isOpen: [{ type: Input }], onClose: [{ type: Output }], onOpen: [{ type: Output }], isOpenChange: [{ type: Output }], id: [{ type: Input }], aiText: [{ type: Input }], textLabel: [{ type: Input }], kind: [{ type: Input }], size: [{ type: Input }], revertActive: [{ type: Input }], revertLabel: [{ type: Input }], ariaLabel: [{ type: Input }], revertClick: [{ type: Output }], aiLabelPopover: [{ type: ViewChild, args: ["aiLabelPopoverHost", { read: AILabelPopoverDirective }] }], hostkeys: [{ type: HostListener, args: ["keyup", ["$event"]] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ai-label.component.js","sourceRoot":"","sources":["../../../src/ai-label/ai-label.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n\tAfterViewInit,\r\n\tChangeDetectionStrategy,\r\n\tComponent,\r\n\tElementRef,\r\n\tEventEmitter,\r\n\tHostBinding,\r\n\tHostListener,\r\n\tInput,\r\n\tOnChanges,\r\n\tOnDestroy,\r\n\tOutput,\r\n\tSimpleChanges,\r\n\tViewChild\r\n} from \"@angular/core\";\r\nimport { Placement } from \"@floating-ui/dom\";\r\n\r\nimport { AILabelPopoverDirective } from \"./ai-label-popover.directive\";\r\n\r\n/**\r\n * @deprecated alignments — use `Placement` names\r\n */\r\ntype DeprecatedAILabelAlign =\r\n\t| \"top-left\"\r\n\t| \"top-right\"\r\n\t| \"bottom-left\"\r\n\t| \"bottom-right\"\r\n\t| \"left-bottom\"\r\n\t| \"left-top\"\r\n\t| \"right-bottom\"\r\n\t| \"right-top\";\r\n\r\n/**\r\n * AI-branded toggletip control (`cds-ai-label`). Renders an \"AI\" badge that opens a\r\n * popover; projected content and optional actions use `ng-content`.\r\n *\r\n * Get started with importing the module:\r\n *\r\n * ```typescript\r\n * import { AILabelModule } from 'carbon-components-angular';\r\n * ```\r\n *\r\n * ```html\r\n * <cds-ai-label size=\"md\">\r\n *   <div>\r\n *     <p>AI Explained</p>\r\n *     <h2>84%</h2>\r\n *     <p>Confidence score</p>\r\n *   </div>\r\n *   <div cdsAILabelActions>\r\n *     <button cdsButton=\"ghost\" size=\"sm\">View details</button>\r\n *   </div>\r\n * </cds-ai-label>\r\n * ```\r\n *\r\n * `[cdsAILabelActions]` adds `cds--toggletip-actions` and `cds--ai-label-actions`\r\n * to its host. Place it as a **sibling** of the body content, both direct\r\n * children of `<cds-ai-label>`. `[cdsAILabelContent]` is an optional marker; the\r\n * `cds--ai-label-content` / `cds--toggletip-content` classes come from this\r\n * component’s template.\r\n *\r\n * [See demo](../../?path=/story/components-ai-label--default)\r\n */\r\n@Component({\r\n\tselector: \"cds-ai-label, ibm-ai-label\",\r\n\tchangeDetection: ChangeDetectionStrategy.OnPush,\r\n\ttemplate: `\r\n\t\t<ng-container *ngIf=\"!revertActive\">\r\n\t\t\t<span\r\n\t\t\t\t#aiLabelPopoverHost\r\n\t\t\t\tcdsAILabelPopover\r\n\t\t\t\tclass=\"cds--toggletip\"\r\n\t\t\t\t[isOpen]=\"isOpen\"\r\n\t\t\t\t(isOpenChange)=\"onPopoverIsOpenChange($event)\"\r\n\t\t\t\t(onOpen)=\"onOpen.emit($event)\"\r\n\t\t\t\t(onClose)=\"onClose.emit($event)\"\r\n\t\t\t\t[align]=\"align\"\r\n\t\t\t\t[caret]=\"caret\"\r\n\t\t\t\t[dropShadow]=\"dropShadow\"\r\n\t\t\t\t[highContrast]=\"highContrast\"\r\n\t\t\t\t[autoAlign]=\"autoAlign\"\r\n\t\t\t\t[alignmentAxisOffset]=\"alignmentAxisOffset\">\r\n\t\t\t\t<button\r\n\t\t\t\t\ttype=\"button\"\r\n\t\t\t\t\t[attr.aria-label]=\"computedAriaLabel\"\r\n\t\t\t\t\t[attr.aria-expanded]=\"isOpen\"\r\n\t\t\t\t\t[attr.aria-controls]=\"id\"\r\n\t\t\t\t\t[ngClass]=\"triggerClasses\"\r\n\t\t\t\t\t(click)=\"onTriggerClick($event)\">\r\n\t\t\t\t\t<span class=\"cds--ai-label__text\">{{aiText}}</span>\r\n\t\t\t\t\t<span *ngIf=\"kind === 'inline' && textLabel\" class=\"cds--ai-label__additional-text\">{{textLabel}}</span>\r\n\t\t\t\t</button>\r\n\r\n\t\t\t\t<span\r\n\t\t\t\t\t[id]=\"id\"\r\n\t\t\t\t\tclass=\"cds--popover\"\r\n\t\t\t\t\taria-live=\"polite\">\r\n\t\t\t\t\t<span class=\"cds--popover-content cds--ai-label-content\">\r\n\t\t\t\t\t\t<div class=\"cds--toggletip-content\">\r\n\t\t\t\t\t\t\t<ng-content></ng-content>\r\n\t\t\t\t\t\t</div>\r\n\t\t\t\t\t\t<span *ngIf=\"autoAlign\" class=\"cds--popover-caret cds--popover--auto-align\"></span>\r\n\t\t\t\t\t</span>\r\n\t\t\t\t\t<span *ngIf=\"!autoAlign\" class=\"cds--popover-caret\"></span>\r\n\t\t\t\t</span>\r\n\t\t\t</span>\r\n\t\t</ng-container>\r\n\r\n\t\t<cds-icon-button\r\n\t\t\t*ngIf=\"revertActive\"\r\n\t\t\tkind=\"ghost\"\r\n\t\t\tsize=\"sm\"\r\n\t\t\t[description]=\"revertLabel\"\r\n\t\t\t[autoAlign]=\"autoAlign\"\r\n\t\t\t[buttonAttributes]=\"{ 'aria-label': revertLabel }\"\r\n\t\t\t(click)=\"onRevertButtonClick($event)\">\r\n\t\t\t<svg cdsIcon=\"undo\" size=\"16\"></svg>\r\n\t\t</cds-icon-button>\r\n\t`\r\n})\r\nexport class AILabelComponent implements AfterViewInit, OnChanges, OnDestroy {\r\n\tstatic labelCounter = 0;\r\n\r\n\t@HostBinding(\"class.cds--ai-label\") aiLabelClass = true;\r\n\t@HostBinding(\"class.cds--ai-label--revert\") get revertClass() {\r\n\t\treturn this.revertActive;\r\n\t}\r\n\r\n\t/**\r\n\t * Set alignment of popover. Deprecated Carbon alignments are mapped to\r\n\t * floating-ui placements.\r\n\t */\r\n\t@Input() align: DeprecatedAILabelAlign | Placement;\r\n\r\n\t/**\r\n\t * Show caret at the alignment position.\r\n\t */\r\n\t@Input() caret = true;\r\n\r\n\t/**\r\n\t * Enable drop shadow around the popover container.\r\n\t */\r\n\t@Input() dropShadow = false;\r\n\r\n\t/**\r\n\t * Enable high contrast for popover container.\r\n\t */\r\n\t@Input() highContrast = true;\r\n\r\n\t/**\r\n\t * **Experimental**: Use floating-ui to position the tooltip.\r\n\t */\r\n\t@Input() autoAlign = false;\r\n\r\n\t/**\r\n\t * Whether the callout is open.\r\n\t */\r\n\t@Input() isOpen = false;\r\n\r\n\t/**\r\n\t * Emits when the callout is closed.\r\n\t */\r\n\t@Output() onClose = new EventEmitter<Event>();\r\n\r\n\t/**\r\n\t * Emits when the callout is opened.\r\n\t */\r\n\t@Output() onOpen = new EventEmitter<Event>();\r\n\r\n\t/**\r\n\t * Emits when `isOpen` changes (two-way binding).\r\n\t */\r\n\t@Output() isOpenChange = new EventEmitter<boolean>();\r\n\r\n\t/**\r\n\t * Unique id used to associate the trigger button with the popover panel\r\n\t * via `aria-controls` / `id`.\r\n\t */\r\n\t@Input() id = `ai-label-${AILabelComponent.labelCounter++}`;\r\n\r\n\t/**\r\n\t * Text inside the AI badge.\r\n\t */\r\n\t@Input() aiText = \"AI\";\r\n\r\n\t/**\r\n\t * Extra text beside the badge when `kind` is `\"inline\"`.\r\n\t */\r\n\t@Input() textLabel: string;\r\n\r\n\t/**\r\n\t * Set badge shape: `\"default\"` (circular) or `\"inline\"` (pill, optional `textLabel`).\r\n\t */\r\n\t@Input() kind: \"default\" | \"inline\" = \"default\";\r\n\r\n\t/**\r\n\t * Set badge size\r\n\t */\r\n\t@Input() size: \"mini\" | \"2xs\" | \"xs\" | \"sm\" | \"md\" | \"lg\" | \"xl\" = \"xs\";\r\n\r\n\t/**\r\n\t * Horizontal shift along the alignment axis when `autoAlign` is on, matching\r\n\t * React `AILabel` (`alignmentAxisOffset={isSmallIcon ? -24 : 0}` on `Toggletip`).\r\n\t */\r\n\tget alignmentAxisOffset(): number {\r\n\t\treturn [\"mini\", \"2xs\", \"xs\"].includes(this.size) ? -24 : 0;\r\n\t}\r\n\r\n\t/**\r\n\t * When `true`, shows the revert icon instead of the badge (AI-generated value\r\n\t * is active and can be reverted).\r\n\t */\r\n\t@Input() revertActive = false;\r\n\r\n\t/**\r\n\t * Accessible label / tooltip for the revert icon button.\r\n\t */\r\n\t@Input() revertLabel = \"Revert to AI input\";\r\n\r\n\t/**\r\n\t * `aria-label` for the AI badge trigger (combined with `aiText` in `computedAriaLabel`).\r\n\t */\r\n\t@Input() ariaLabel = \"Show information\";\r\n\r\n\t/**\r\n\t * Emitted when the revert icon is clicked.\r\n\t */\r\n\t@Output() revertClick = new EventEmitter<MouseEvent>();\r\n\r\n\t@ViewChild(\"aiLabelPopoverHost\", { read: AILabelPopoverDirective })\r\n\tprivate aiLabelPopover: AILabelPopoverDirective;\r\n\r\n\tprivate readonly documentClick = this.handleOutsideClick.bind(this);\r\n\r\n\tconstructor(private elementRef: ElementRef) {}\r\n\r\n\tonPopoverIsOpenChange(open: boolean): void {\r\n\t\tthis.isOpen = open;\r\n\t\tthis.isOpenChange.emit(open);\r\n\t}\r\n\r\n\tget triggerClasses(): Record<string, boolean> {\r\n\t\treturn {\r\n\t\t\t\"cds--toggletip-button\": true,\r\n\t\t\t\"cds--ai-label__button\": true,\r\n\t\t\t[`cds--ai-label__button--${this.size}`]: true,\r\n\t\t\t[`cds--ai-label__button--${this.kind}`]: true,\r\n\t\t\t\"cds--ai-label__button--inline-with-content\": this.kind === \"inline\" && !!this.textLabel\r\n\t\t};\r\n\t}\r\n\r\n\t/**\r\n\t * Trigger `aria-label`: `\"${aiText} ${ariaLabel}\"`, or\r\n\t * `\"${aiText} ${textLabel}\"` when `kind` is `\"inline\"` and `textLabel` is set.\r\n\t */\r\n\tget computedAriaLabel(): string {\r\n\t\tconst suffix = (this.kind === \"inline\" && this.textLabel) ? this.textLabel : this.ariaLabel;\r\n\t\treturn `${this.aiText} ${suffix}`;\r\n\t}\r\n\r\n\tngAfterViewInit(): void {\r\n\t\tif (this.isOpen) {\r\n\t\t\tdocument.addEventListener(\"click\", this.documentClick);\r\n\t\t}\r\n\t}\r\n\r\n\tngOnChanges(changes: SimpleChanges): void {\r\n\t\tif (changes.revertActive && !changes.revertActive.firstChange && changes.revertActive.currentValue) {\r\n\t\t\tthis.isOpen = false;\r\n\t\t\tdocument.removeEventListener(\"click\", this.documentClick);\r\n\t\t}\r\n\t}\r\n\r\n\tngOnDestroy(): void {\r\n\t\tdocument.removeEventListener(\"click\", this.documentClick);\r\n\t}\r\n\r\n\tonTriggerClick(event: MouseEvent): void {\r\n\t\tconst opening = !this.isOpen;\r\n\t\tif (opening) {\r\n\t\t\tdocument.addEventListener(\"click\", this.documentClick);\r\n\t\t} else {\r\n\t\t\tdocument.removeEventListener(\"click\", this.documentClick);\r\n\t\t}\r\n\t\tthis.aiLabelPopover?.handleChange(opening, event);\r\n\t}\r\n\r\n\tonRevertButtonClick(event: MouseEvent): void {\r\n\t\tthis.revertClick.emit(event);\r\n\t}\r\n\r\n\t@HostListener(\"keyup\", [\"$event\"])\r\n\thostkeys(event: KeyboardEvent): void {\r\n\t\tif (this.isOpen && event.key === \"Escape\") {\r\n\t\t\tevent.stopPropagation();\r\n\t\t\tdocument.removeEventListener(\"click\", this.documentClick);\r\n\t\t\tthis.aiLabelPopover?.handleChange(false, event);\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Dismisses the popover when a click lands outside the host element.\r\n\t */\r\n\tprivate handleOutsideClick(event: MouseEvent): void {\r\n\t\tif (!this.elementRef.nativeElement.contains(event.target as Node)) {\r\n\t\t\tthis.aiLabelPopover?.handleChange(false, event);\r\n\t\t\tdocument.removeEventListener(\"click\", this.documentClick);\r\n\t\t}\r\n\t}\r\n}\r\n"]}