@hxui/angular
Version:
An Angular library based on the [HXUI design system](https://hxui.io).
270 lines • 34.2 kB
JavaScript
import { Directionality } from '@angular/cdk/bidi';
import { Overlay, ScrollDispatcher } from '@angular/cdk/overlay';
import { ComponentPortal } from '@angular/cdk/portal';
import { ComponentFactoryResolver, ContentChild, Directive, ElementRef, HostBinding, HostListener, Input, NgZone, Optional, ViewContainerRef } from '@angular/core';
import { Subject } from 'rxjs';
import { take, takeUntil } from 'rxjs/operators';
import { Context } from '../enums';
import { TooltipContentComponent } from './tooltip-content.component';
import { TooltipDynamicContentDirective } from './tooltip-dynamic-content.directive';
import { TooltipConfig } from './tooltip.config';
import * as i0 from "@angular/core";
import * as i1 from "@angular/cdk/overlay";
import * as i2 from "./tooltip.config";
import * as i3 from "@angular/cdk/bidi";
export class TooltipDirective {
constructor(_elementRef, _viewContainerRef, overlay, _ngZone, _scrollDispatcher, _componentFactoryResolver, _config, _dir) {
this._elementRef = _elementRef;
this._viewContainerRef = _viewContainerRef;
this.overlay = overlay;
this._ngZone = _ngZone;
this._scrollDispatcher = _scrollDispatcher;
this._componentFactoryResolver = _componentFactoryResolver;
this._config = _config;
this._dir = _dir;
this._destroyed = new Subject();
this.position = 'relative';
this.placement = 'bottom';
this.showDelay = this._config.showDelay;
this.hideDelay = this._config.hideDelay;
this.context = Context.None;
this.maxWidth = 200;
this.autoClose = true;
Object.assign(this, _config);
}
show() {
this._show();
}
close() {
if (this.autoClose) {
this._hide();
}
}
/**
* Dispose the tooltip when destroyed.
*/
ngOnDestroy() {
if (this._overlayRef) {
this._overlayRef.dispose();
this._tooltipInstance = null;
}
this._destroyed.next(true);
this._destroyed.complete();
}
hide() {
this._hide();
}
_show(delay = this.showDelay) {
if (this.disabled || (!this.content && !this.dynamicContent)) {
return;
}
const overlayRef = this._createOverlay();
this._detach();
this._portal =
this._portal ||
new ComponentPortal(TooltipContentComponent, this._viewContainerRef);
this._tooltipInstance = overlayRef.attach(this._portal).instance;
this._tooltipInstance
.afterHidden()
.pipe(takeUntil(this._destroyed))
.subscribe(() => this._detach());
this._updateTooltipContent();
this._tooltipInstance.show(delay);
}
_hide(delay = this.hideDelay) {
if (this._tooltipInstance) {
this._tooltipInstance.hide(delay);
}
}
_createOverlay() {
if (this._overlayRef) {
return this._overlayRef;
}
const positionStrategy = this.overlay
.position()
.flexibleConnectedTo(this._elementRef)
.withTransformOriginOn('.hx-tooltip')
.withFlexibleDimensions(false);
this._overlayRef = this.overlay.create({
positionStrategy: positionStrategy,
panelClass: 'hxa-tooltip-panel',
scrollStrategy: this.overlay.scrollStrategies.reposition(),
hasBackdrop: !this.autoClose,
backdropClass: 'cdk-overlay-transparent-backdrop'
});
this._updatePosition();
this._overlayRef
.detachments()
.pipe(takeUntil(this._destroyed))
.subscribe(() => this._detach());
this._overlayRef.backdropClick().subscribe(() => this._hide());
const position = this._overlayRef.getConfig()
.positionStrategy;
position.positionChanges.pipe(takeUntil(this._destroyed)).subscribe(pos => {
if (pos.connectionPair.originX === 'start') {
this.placement = 'left';
}
else if (pos.connectionPair.originX === 'end') {
this.placement = 'right';
}
this._updateTooltipContent();
});
return this._overlayRef;
}
_updatePosition() {
const position = this._overlayRef.getConfig()
.positionStrategy;
const origin = this._getOrigin();
const overlay = this._getOverlayPosition();
position.withPositions([
{ ...origin.main, ...overlay.main },
{ ...origin.fallback, ...overlay.fallback }
]);
}
/**
* Returns the origin position and a fallback position based on the user's position preference.
* The fallback position is the inverse of the origin (e.g. `'bottom' -> 'top'`).
*/
_getOrigin() {
const placement = this.placement;
let originPlacement;
if (placement === 'top' || placement === 'bottom') {
originPlacement = {
originX: 'center',
originY: placement === 'top' ? 'top' : 'bottom'
};
}
else if (placement === 'left') {
originPlacement = { originX: 'start', originY: 'center' };
}
else if (placement === 'right') {
originPlacement = { originX: 'end', originY: 'center' };
}
else {
console.error('Position error', placement);
}
const { x, y } = this._invertPosition(originPlacement.originX, originPlacement.originY);
return {
main: originPlacement,
fallback: { originX: x, originY: y }
};
}
/** Returns the overlay position and a fallback position based on the user's preference */
_getOverlayPosition() {
const placement = this.placement;
let overlayPlacement;
if (placement === 'top') {
overlayPlacement = { overlayX: 'center', overlayY: 'bottom' };
}
else if (placement === 'bottom') {
overlayPlacement = { overlayX: 'center', overlayY: 'top' };
}
else if (placement === 'left') {
overlayPlacement = { overlayX: 'end', overlayY: 'center' };
}
else if (placement === 'right') {
overlayPlacement = { overlayX: 'start', overlayY: 'center' };
}
else {
console.error('Could not find a position', placement);
}
const { x, y } = this._invertPosition(overlayPlacement.overlayX, overlayPlacement.overlayY);
return {
main: overlayPlacement,
fallback: { overlayX: x, overlayY: y }
};
}
_invertPosition(x, y) {
if (this.position === 'top' || this.position === 'bottom') {
if (y === 'top') {
y = 'bottom';
}
else if (y === 'bottom') {
y = 'top';
}
}
else {
if (x === 'end') {
x = 'start';
}
else if (x === 'start') {
x = 'end';
}
}
return { x, y };
}
_detach() {
if (this._overlayRef && this._overlayRef.hasAttached()) {
this._overlayRef.detach();
}
this._tooltipInstance = null;
}
/** Updates the tooltip content and repositions the overlay according to the new content length */
_updateTooltipContent() {
// Must wait for the content to be painted to the tooltip so that the overlay can properly
// calculate the correct positioning based on the size of its contents.
if (this._tooltipInstance) {
this._tooltipInstance.content = this.content;
this._tooltipInstance.placement = this.placement;
this._tooltipInstance.context = this.context;
this._tooltipInstance.maxWidth = this.maxWidth;
if (this.dynamicContent) {
this._tooltipInstance.dynamicContent = this.dynamicContent.templateRef;
}
this._ngZone.onMicrotaskEmpty
.asObservable()
.pipe(take(1), takeUntil(this._destroyed))
.subscribe(() => {
if (this._tooltipInstance) {
this._overlayRef.updatePosition();
}
});
}
}
}
TooltipDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: TooltipDirective, deps: [{ token: i0.ElementRef }, { token: i0.ViewContainerRef }, { token: i1.Overlay }, { token: i0.NgZone }, { token: i1.ScrollDispatcher }, { token: i0.ComponentFactoryResolver }, { token: i2.TooltipConfig }, { token: i3.Directionality, optional: true }], target: i0.ɵɵFactoryTarget.Directive });
TooltipDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.11", type: TooltipDirective, selector: "[hxTooltip], [hxaTooltip]", inputs: { content: ["hxTooltip", "content"], disabled: "disabled", placement: "placement", showDelay: "showDelay", hideDelay: "hideDelay", context: "context", maxWidth: "maxWidth", autoClose: "autoClose" }, host: { listeners: { "focusin": "show()", "mouseenter": "show()", "focusout": "close()", "mouseleave": "close()" }, properties: { "style.position": "this.position" } }, queries: [{ propertyName: "dynamicContent", first: true, predicate: TooltipDynamicContentDirective, descendants: true }], ngImport: i0 });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: TooltipDirective, decorators: [{
type: Directive,
args: [{
selector: '[hxTooltip], [hxaTooltip]'
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ViewContainerRef }, { type: i1.Overlay }, { type: i0.NgZone }, { type: i1.ScrollDispatcher }, { type: i0.ComponentFactoryResolver }, { type: i2.TooltipConfig }, { type: i3.Directionality, decorators: [{
type: Optional
}] }]; }, propDecorators: { dynamicContent: [{
type: ContentChild,
args: [TooltipDynamicContentDirective]
}], position: [{
type: HostBinding,
args: ['style.position']
}], content: [{
type: Input,
args: ['hxTooltip']
}], disabled: [{
type: Input
}], placement: [{
type: Input
}], showDelay: [{
type: Input
}], hideDelay: [{
type: Input
}], context: [{
type: Input
}], maxWidth: [{
type: Input
}], autoClose: [{
type: Input
}], show: [{
type: HostListener,
args: ['focusin']
}, {
type: HostListener,
args: ['mouseenter']
}], close: [{
type: HostListener,
args: ['focusout']
}, {
type: HostListener,
args: ['mouseleave']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../projects/hx-ui/src/lib/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directionality } from '@angular/cdk/bidi';\r\nimport {\r\n  FlexibleConnectedPositionStrategy,\r\n  HorizontalConnectionPos,\r\n  OriginConnectionPosition,\r\n  Overlay,\r\n  OverlayConnectionPosition,\r\n  OverlayRef,\r\n  ScrollDispatcher,\r\n  VerticalConnectionPos\r\n} from '@angular/cdk/overlay';\r\nimport { ComponentPortal } from '@angular/cdk/portal';\r\nimport {\r\n  ComponentFactoryResolver,\r\n  ContentChild,\r\n  Directive,\r\n  ElementRef,\r\n  HostBinding,\r\n  HostListener,\r\n  Input,\r\n  NgZone,\r\n  OnDestroy,\r\n  Optional,\r\n  ViewContainerRef\r\n} from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { take, takeUntil } from 'rxjs/operators';\r\nimport { Context } from '../enums';\r\nimport { TooltipContentComponent } from './tooltip-content.component';\r\nimport { TooltipDynamicContentDirective } from './tooltip-dynamic-content.directive';\r\nimport { TooltipConfig } from './tooltip.config';\r\n\r\n@Directive({\r\n  selector: '[hxTooltip], [hxaTooltip]'\r\n})\r\nexport class TooltipDirective implements OnDestroy {\r\n  @ContentChild(TooltipDynamicContentDirective)\r\n  dynamicContent: TooltipDynamicContentDirective;\r\n\r\n  _overlayRef: OverlayRef | null;\r\n  _tooltipInstance: TooltipContentComponent | null;\r\n\r\n  private _portal: ComponentPortal<TooltipContentComponent>;\r\n  private readonly _destroyed = new Subject();\r\n\r\n  @HostBinding('style.position')\r\n  position = 'relative';\r\n\r\n  @Input('hxTooltip')\r\n  content: string;\r\n\r\n  @Input()\r\n  disabled: boolean;\r\n\r\n  @Input()\r\n  placement: 'top' | 'bottom' | 'left' | 'right' = 'bottom';\r\n\r\n  @Input()\r\n  showDelay = this._config.showDelay;\r\n\r\n  @Input()\r\n  hideDelay = this._config.hideDelay;\r\n\r\n  @Input()\r\n  context: Context = Context.None;\r\n\r\n  @Input()\r\n  maxWidth = 200;\r\n\r\n  @Input()\r\n  autoClose = true;\r\n\r\n  @HostListener('focusin')\r\n  @HostListener('mouseenter')\r\n  show() {\r\n    this._show();\r\n  }\r\n\r\n  @HostListener('focusout')\r\n  @HostListener('mouseleave')\r\n  close() {\r\n    if (this.autoClose) {\r\n      this._hide();\r\n    }\r\n  }\r\n\r\n  constructor(\r\n    private _elementRef: ElementRef,\r\n    private _viewContainerRef: ViewContainerRef,\r\n    public overlay: Overlay,\r\n    private _ngZone: NgZone,\r\n    private _scrollDispatcher: ScrollDispatcher,\r\n    private _componentFactoryResolver: ComponentFactoryResolver,\r\n    private _config: TooltipConfig,\r\n    @Optional() private _dir: Directionality\r\n  ) {\r\n    Object.assign(this, _config);\r\n  }\r\n\r\n  /**\r\n   * Dispose the tooltip when destroyed.\r\n   */\r\n  ngOnDestroy() {\r\n    if (this._overlayRef) {\r\n      this._overlayRef.dispose();\r\n      this._tooltipInstance = null;\r\n    }\r\n\r\n    this._destroyed.next(true);\r\n    this._destroyed.complete();\r\n  }\r\n\r\n  hide() {\r\n    this._hide();\r\n  }\r\n\r\n  private _show(delay: number = this.showDelay) {\r\n    if (this.disabled || (!this.content && !this.dynamicContent)) {\r\n      return;\r\n    }\r\n\r\n    const overlayRef = this._createOverlay();\r\n\r\n    this._detach();\r\n    this._portal =\r\n      this._portal ||\r\n      new ComponentPortal(TooltipContentComponent, this._viewContainerRef);\r\n    this._tooltipInstance = overlayRef.attach(this._portal).instance;\r\n    this._tooltipInstance\r\n      .afterHidden()\r\n      .pipe(takeUntil(this._destroyed))\r\n      .subscribe(() => this._detach());\r\n\r\n    this._updateTooltipContent();\r\n    this._tooltipInstance!.show(delay);\r\n  }\r\n\r\n  private _hide(delay: number = this.hideDelay) {\r\n    if (this._tooltipInstance) {\r\n      this._tooltipInstance.hide(delay);\r\n    }\r\n  }\r\n\r\n  private _createOverlay(): OverlayRef {\r\n    if (this._overlayRef) {\r\n      return this._overlayRef;\r\n    }\r\n\r\n    const positionStrategy = this.overlay\r\n      .position()\r\n      .flexibleConnectedTo(this._elementRef)\r\n      .withTransformOriginOn('.hx-tooltip')\r\n      .withFlexibleDimensions(false);\r\n\r\n    this._overlayRef = this.overlay.create({\r\n      positionStrategy: positionStrategy,\r\n      panelClass: 'hxa-tooltip-panel',\r\n      scrollStrategy: this.overlay.scrollStrategies.reposition(),\r\n      hasBackdrop: !this.autoClose,\r\n      backdropClass: 'cdk-overlay-transparent-backdrop'\r\n    });\r\n\r\n    this._updatePosition();\r\n\r\n    this._overlayRef\r\n      .detachments()\r\n      .pipe(takeUntil(this._destroyed))\r\n      .subscribe(() => this._detach());\r\n\r\n    this._overlayRef.backdropClick().subscribe(() => this._hide());\r\n\r\n    const position = this._overlayRef.getConfig()\r\n      .positionStrategy as FlexibleConnectedPositionStrategy;\r\n    position.positionChanges.pipe(takeUntil(this._destroyed)).subscribe(pos => {\r\n      if (pos.connectionPair.originX === 'start') {\r\n        this.placement = 'left';\r\n      } else if (pos.connectionPair.originX === 'end') {\r\n        this.placement = 'right';\r\n      }\r\n      this._updateTooltipContent();\r\n    });\r\n\r\n    return this._overlayRef;\r\n  }\r\n\r\n  private _updatePosition() {\r\n    const position = this._overlayRef!.getConfig()\r\n      .positionStrategy as FlexibleConnectedPositionStrategy;\r\n    const origin = this._getOrigin();\r\n    const overlay = this._getOverlayPosition();\r\n\r\n    position.withPositions([\r\n      { ...origin.main, ...overlay.main },\r\n      { ...origin.fallback, ...overlay.fallback }\r\n    ]);\r\n  }\r\n\r\n  /**\r\n   * Returns the origin position and a fallback position based on the user's position preference.\r\n   * The fallback position is the inverse of the origin (e.g. `'bottom' -> 'top'`).\r\n   */\r\n  private _getOrigin(): {\r\n    main: OriginConnectionPosition;\r\n    fallback: OriginConnectionPosition;\r\n  } {\r\n    const placement = this.placement;\r\n    let originPlacement: OriginConnectionPosition;\r\n\r\n    if (placement === 'top' || placement === 'bottom') {\r\n      originPlacement = {\r\n        originX: 'center',\r\n        originY: placement === 'top' ? 'top' : 'bottom'\r\n      };\r\n    } else if (placement === 'left') {\r\n      originPlacement = { originX: 'start', originY: 'center' };\r\n    } else if (placement === 'right') {\r\n      originPlacement = { originX: 'end', originY: 'center' };\r\n    } else {\r\n      console.error('Position error', placement);\r\n    }\r\n\r\n    const { x, y } = this._invertPosition(\r\n      originPlacement.originX,\r\n      originPlacement.originY\r\n    );\r\n\r\n    return {\r\n      main: originPlacement,\r\n      fallback: { originX: x, originY: y }\r\n    };\r\n  }\r\n\r\n  /** Returns the overlay position and a fallback position based on the user's preference */\r\n  private _getOverlayPosition(): {\r\n    main: OverlayConnectionPosition;\r\n    fallback: OverlayConnectionPosition;\r\n  } {\r\n    const placement = this.placement;\r\n    let overlayPlacement: OverlayConnectionPosition;\r\n\r\n    if (placement === 'top') {\r\n      overlayPlacement = { overlayX: 'center', overlayY: 'bottom' };\r\n    } else if (placement === 'bottom') {\r\n      overlayPlacement = { overlayX: 'center', overlayY: 'top' };\r\n    } else if (placement === 'left') {\r\n      overlayPlacement = { overlayX: 'end', overlayY: 'center' };\r\n    } else if (placement === 'right') {\r\n      overlayPlacement = { overlayX: 'start', overlayY: 'center' };\r\n    } else {\r\n      console.error('Could not find a position', placement);\r\n    }\r\n\r\n    const { x, y } = this._invertPosition(\r\n      overlayPlacement.overlayX,\r\n      overlayPlacement.overlayY\r\n    );\r\n\r\n    return {\r\n      main: overlayPlacement,\r\n      fallback: { overlayX: x, overlayY: y }\r\n    };\r\n  }\r\n\r\n  private _invertPosition(\r\n    x: HorizontalConnectionPos,\r\n    y: VerticalConnectionPos\r\n  ) {\r\n    if (this.position === 'top' || this.position === 'bottom') {\r\n      if (y === 'top') {\r\n        y = 'bottom';\r\n      } else if (y === 'bottom') {\r\n        y = 'top';\r\n      }\r\n    } else {\r\n      if (x === 'end') {\r\n        x = 'start';\r\n      } else if (x === 'start') {\r\n        x = 'end';\r\n      }\r\n    }\r\n\r\n    return { x, y };\r\n  }\r\n\r\n  private _detach() {\r\n    if (this._overlayRef && this._overlayRef.hasAttached()) {\r\n      this._overlayRef.detach();\r\n    }\r\n    this._tooltipInstance = null;\r\n  }\r\n\r\n  /** Updates the tooltip content and repositions the overlay according to the new content length */\r\n  private _updateTooltipContent() {\r\n    // Must wait for the content to be painted to the tooltip so that the overlay can properly\r\n    // calculate the correct positioning based on the size of its contents.\r\n    if (this._tooltipInstance) {\r\n      this._tooltipInstance.content = this.content;\r\n      this._tooltipInstance.placement = this.placement;\r\n      this._tooltipInstance.context = this.context;\r\n      this._tooltipInstance.maxWidth = this.maxWidth;\r\n      if (this.dynamicContent) {\r\n        this._tooltipInstance.dynamicContent = this.dynamicContent.templateRef;\r\n      }\r\n\r\n      this._ngZone.onMicrotaskEmpty\r\n        .asObservable()\r\n        .pipe(take(1), takeUntil(this._destroyed))\r\n        .subscribe(() => {\r\n          if (this._tooltipInstance) {\r\n            this._overlayRef!.updatePosition();\r\n          }\r\n        });\r\n    }\r\n  }\r\n}\r\n"]}