carbon-components-angular
Version:
Next generation components
349 lines (345 loc) • 32.4 kB
JavaScript
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"]}