@eternalheart/ngx-file-preview
Version:
A powerful Angular file preview component library supporting multiple file formats including images, videos, PDFs, Office documents, text files and more.
125 lines • 21.4 kB
JavaScript
import { Component, Directive, HostListener, Input } from "@angular/core";
import * as i0 from "@angular/core";
import * as i1 from "../services";
export class TooltipDirective {
constructor(el, renderer, viewContainer, previewService) {
this.el = el;
this.renderer = renderer;
this.viewContainer = viewContainer;
this.previewService = previewService;
this.delay = 500;
this.positions = ['top', 'bottom', 'left', 'right'];
this.currentPosition = 'top';
}
onMouseEnter() {
this.clearTimers();
this.showTimeout = setTimeout(() => this.show(), this.delay);
}
onMouseLeave() {
this.clearTimers();
this.hideTimeout = setTimeout(() => this.hide(), 100);
}
show() {
if (!this.content)
return;
if (!this.tooltip) {
// 动态创建组件
const factory = this.viewContainer.createComponent(TooltipComponent);
this.tooltip = factory.location.nativeElement;
factory.instance.content = this.content;
// 立即显示内容
this.renderer.addClass(this.tooltip, 'visible');
this.previewService.modalElement?.querySelector('.nfp-modal__overlay').appendChild(this.tooltip);
factory.changeDetectorRef.detectChanges();
}
// 计算最佳位置
const hostRect = this.el.nativeElement.getBoundingClientRect();
const tooltipRect = this.tooltip.getBoundingClientRect();
const viewportWidth = window.innerWidth;
const viewportHeight = window.innerHeight;
// 检查每个位置的可用空间
const spaces = {
top: hostRect.top,
bottom: viewportHeight - (hostRect.bottom),
left: hostRect.left,
right: viewportWidth - (hostRect.right)
};
// 找到最佳位置
this.currentPosition = this.positions.reduce((best, current) => spaces[current] > spaces[best] ? current : best);
// 根据位置设置样式类
this.positions.forEach(pos => this.renderer.removeClass(this.tooltip, pos));
this.renderer.addClass(this.tooltip, this.currentPosition);
// 根据位置计算坐标
let top, left;
switch (this.currentPosition) {
case 'top':
top = hostRect.top - tooltipRect.height - 8;
left = hostRect.left + (hostRect.width - tooltipRect.width) / 2;
break;
case 'bottom':
top = hostRect.bottom + 8;
left = hostRect.left + (hostRect.width - tooltipRect.width) / 2;
break;
case 'left':
top = hostRect.top + (hostRect.height - tooltipRect.height) / 2;
left = hostRect.left - tooltipRect.width - 8;
break;
case 'right':
top = hostRect.top + (hostRect.height - tooltipRect.height) / 2;
left = hostRect.right + 8;
break;
}
// 确保tooltip不超出视口
top = Math.max(8, Math.min(viewportHeight - tooltipRect.height - 8, top));
left = Math.max(8, Math.min(viewportWidth - tooltipRect.width - 8, left));
this.renderer.setStyle(this.tooltip, 'top', `${top}px`);
this.renderer.setStyle(this.tooltip, 'left', `${left}px`);
}
hide() {
if (this.tooltip) {
this.renderer.removeClass(this.tooltip, 'visible');
setTimeout(() => {
this.viewContainer.clear();
this.tooltip = null;
}, 300); // 增加动画时间
}
}
clearTimers() {
clearTimeout(this.showTimeout);
clearTimeout(this.hideTimeout);
}
ngOnDestroy() {
this.viewContainer.clear();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TooltipDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.ViewContainerRef }, { token: i1.PreviewService }], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.12", type: TooltipDirective, isStandalone: true, selector: "[tooltip]", inputs: { content: ["tooltip", "content"], delay: "delay" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()" } }, ngImport: i0 }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TooltipDirective, decorators: [{
type: Directive,
args: [{ selector: '[tooltip]', standalone: true }]
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ViewContainerRef }, { type: i1.PreviewService }], propDecorators: { content: [{
type: Input,
args: ['tooltip']
}], delay: [{
type: Input
}], onMouseEnter: [{
type: HostListener,
args: ['mouseenter']
}], onMouseLeave: [{
type: HostListener,
args: ['mouseleave']
}] } });
export class TooltipComponent {
constructor() {
this.content = "";
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TooltipComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TooltipComponent, isStandalone: true, selector: "ngx-file-tooltip", inputs: { content: "content" }, ngImport: i0, template: `{{ content }}`, isInline: true, styles: [":host{position:absolute;background:#000000d9;color:#fff;font-size:14px;padding:6px 8px;border-radius:2px;box-shadow:0 3px 6px -4px #0000001f,0 6px 16px #00000014,0 9px 28px 8px #0000000d;max-width:250px;min-height:24px;word-wrap:break-word;z-index:999;pointer-events:none;opacity:0;display:flex;justify-content:center;align-items:center;transform:scale(.8);transform-origin:center;transition:opacity .2s ease-in-out,transform .2s ease-in-out}:host.visible{opacity:1;transform:scale(1)}:host:after{content:\"\";position:absolute;width:0;height:0;border:5px solid transparent}:host.top:after{border-top-color:#000000d9;bottom:-10px;left:50%;transform:translate(-50%)}:host.bottom:after{border-bottom-color:#000000d9;top:-10px;left:50%;transform:translate(-50%)}:host.left:after{border-left-color:#000000d9;right:-10px;top:50%;transform:translateY(-50%)}:host.right:after{border-right-color:#000000d9;left:-10px;top:50%;transform:translateY(-50%)}\n"] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TooltipComponent, decorators: [{
type: Component,
args: [{ selector: 'ngx-file-tooltip', template: `{{ content }}`, standalone: true, styles: [":host{position:absolute;background:#000000d9;color:#fff;font-size:14px;padding:6px 8px;border-radius:2px;box-shadow:0 3px 6px -4px #0000001f,0 6px 16px #00000014,0 9px 28px 8px #0000000d;max-width:250px;min-height:24px;word-wrap:break-word;z-index:999;pointer-events:none;opacity:0;display:flex;justify-content:center;align-items:center;transform:scale(.8);transform-origin:center;transition:opacity .2s ease-in-out,transform .2s ease-in-out}:host.visible{opacity:1;transform:scale(1)}:host:after{content:\"\";position:absolute;width:0;height:0;border:5px solid transparent}:host.top:after{border-top-color:#000000d9;bottom:-10px;left:50%;transform:translate(-50%)}:host.bottom:after{border-bottom-color:#000000d9;top:-10px;left:50%;transform:translate(-50%)}:host.left:after{border-left-color:#000000d9;right:-10px;top:50%;transform:translateY(-50%)}:host.right:after{border-right-color:#000000d9;left:-10px;top:50%;transform:translateY(-50%)}\n"] }]
}], propDecorators: { content: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.diretive.js","sourceRoot":"","sources":["../../../../../libs/ngx-file-preview/src/lib/directives/tooltip.diretive.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,SAAS,EAET,YAAY,EACZ,KAAK,EAIN,MAAM,eAAe,CAAC;;;AAIvB,MAAM,OAAO,gBAAgB;IAU3B,YACU,EAAc,EACd,QAAmB,EACnB,aAA+B,EAC/B,cAA6B;QAH7B,OAAE,GAAF,EAAE,CAAY;QACd,aAAQ,GAAR,QAAQ,CAAW;QACnB,kBAAa,GAAb,aAAa,CAAkB;QAC/B,mBAAc,GAAd,cAAc,CAAe;QAZ9B,UAAK,GAAW,GAAG,CAAC;QAKrB,cAAS,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;QAC/C,oBAAe,GAAG,KAAK,CAAC;IAO7B,CAAC;IAEwB,YAAY;QACtC,IAAI,CAAC,WAAW,EAAE,CAAC;QACnB,IAAI,CAAC,WAAW,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/D,CAAC;IAE2B,YAAY;QACtC,IAAI,CAAC,WAAW,EAAE,CAAC;QACnB,IAAI,CAAC,WAAW,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,GAAG,CAAC,CAAC;IACxD,CAAC;IAEO,IAAI;QACV,IAAG,CAAC,IAAI,CAAC,OAAO;YAAC,OAAO;QACxB,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YAClB,SAAS;YACT,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;YACrE,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC;YAC9C,OAAO,CAAC,QAAQ,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC;YACxC,SAAS;YACT,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;YAChD,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,aAAa,CAAC,qBAAqB,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACjG,OAAO,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAA;QAC3C,CAAC;QAED,SAAS;QACT,MAAM,QAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC;QAC/D,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,qBAAqB,EAAE,CAAC;QACzD,MAAM,aAAa,GAAG,MAAM,CAAC,UAAU,CAAC;QACxC,MAAM,cAAc,GAAG,MAAM,CAAC,WAAW,CAAC;QAE1C,cAAc;QACd,MAAM,MAAM,GAA2B;YACrC,GAAG,EAAE,QAAQ,CAAC,GAAG;YACjB,MAAM,EAAE,cAAc,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC;YAC1C,IAAI,EAAE,QAAQ,CAAC,IAAI;YACnB,KAAK,EAAE,aAAa,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC;SACxC,CAAC;QAEF,SAAS;QACT,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,OAAO,EAAE,EAAE,CAC7D,MAAM,CAAC,OAAO,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAChD,CAAC;QAEF,YAAY;QACZ,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5E,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC;QAE3D,WAAW;QACX,IAAI,GAAG,EAAE,IAAI,CAAC;QACd,QAAO,IAAI,CAAC,eAAe,EAAE,CAAC;YAC5B,KAAK,KAAK;gBACR,GAAG,GAAG,QAAQ,CAAC,GAAG,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;gBAC5C,IAAI,GAAG,QAAQ,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAChE,MAAM;YACR,KAAK,QAAQ;gBACX,GAAG,GAAG,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;gBAC1B,IAAI,GAAG,QAAQ,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAChE,MAAM;YACR,KAAK,MAAM;gBACT,GAAG,GAAG,QAAQ,CAAC,GAAG,GAAG,CAAC,QAAQ,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;gBAChE,IAAI,GAAG,QAAQ,CAAC,IAAI,GAAG,WAAW,CAAC,KAAK,GAAG,CAAC,CAAC;gBAC7C,MAAM;YACR,KAAK,OAAO;gBACV,GAAG,GAAG,QAAQ,CAAC,GAAG,GAAG,CAAC,QAAQ,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;gBAChE,IAAI,GAAG,QAAQ,CAAC,KAAK,GAAG,CAAC,CAAC;gBAC1B,MAAM;QACV,CAAC;QAED,iBAAiB;QACjB,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,cAAc,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;QAC1E,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,aAAa,GAAG,WAAW,CAAC,KAAK,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;QAE1E,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,EAAE,GAAG,GAAG,IAAI,CAAC,CAAC;QACxD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,IAAI,CAAC,CAAC;IAC5D,CAAC;IAEO,IAAI;QACV,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;YACnD,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAA;gBAC1B,IAAI,CAAC,OAAO,GAAG,IAAK,CAAA;YACtB,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,SAAS;QACpB,CAAC;IACH,CAAC;IAEO,WAAW;QACjB,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC/B,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IACjC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;IAC7B,CAAC;+GA7GU,gBAAgB;mGAAhB,gBAAgB;;4FAAhB,gBAAgB;kBAD5B,SAAS;mBAAC,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAE,IAAI,EAAC;mKAEjC,OAAO;sBAAxB,KAAK;uBAAC,SAAS;gBACP,KAAK;sBAAb,KAAK;gBAesB,YAAY;sBAAvC,YAAY;uBAAC,YAAY;gBAKE,YAAY;sBAAvC,YAAY;uBAAC,YAAY;;AA8J5B,MAAM,OAAO,gBAAgB;IArE7B;QAsEW,YAAO,GAAW,EAAE,CAAC;KAC/B;+GAFY,gBAAgB;mGAAhB,gBAAgB,4GAnEjB,eAAe;;4FAmEd,gBAAgB;kBArE5B,SAAS;+BACE,kBAAkB,YAClB,eAAe,cACb,IAAI;8BAmEP,OAAO;sBAAf,KAAK","sourcesContent":["import {\n  Component,\n  Directive,\n  ElementRef,\n  HostListener,\n  Input,\n  OnDestroy,\n  Renderer2,\n  ViewContainerRef\n} from \"@angular/core\";\nimport {PreviewService} from \"../services\";\n\n@Directive({ selector: '[tooltip]' ,standalone: true})\nexport class TooltipDirective implements OnDestroy {\n  @Input('tooltip') content!: string;\n  @Input() delay: number = 500;\n\n  private tooltip!: HTMLElement;\n  private showTimeout?: any;\n  private hideTimeout?: any;\n  private positions = ['top', 'bottom', 'left', 'right'];\n  private currentPosition = 'top';\n\n  constructor(\n    private el: ElementRef,\n    private renderer: Renderer2,\n    private viewContainer: ViewContainerRef,\n    private previewService:PreviewService\n  ) {}\n\n  @HostListener('mouseenter') onMouseEnter() {\n    this.clearTimers();\n    this.showTimeout = setTimeout(() => this.show(), this.delay);\n  }\n\n  @HostListener('mouseleave') onMouseLeave() {\n    this.clearTimers();\n    this.hideTimeout = setTimeout(() => this.hide(), 100);\n  }\n\n  private show() {\n    if(!this.content)return;\n    if (!this.tooltip) {\n      // 动态创建组件\n      const factory = this.viewContainer.createComponent(TooltipComponent);\n      this.tooltip = factory.location.nativeElement;\n      factory.instance.content = this.content;\n      // 立即显示内容\n      this.renderer.addClass(this.tooltip, 'visible');\n      this.previewService.modalElement?.querySelector('.nfp-modal__overlay').appendChild(this.tooltip);\n      factory.changeDetectorRef.detectChanges()\n    }\n\n    // 计算最佳位置\n    const hostRect = this.el.nativeElement.getBoundingClientRect();\n    const tooltipRect = this.tooltip.getBoundingClientRect();\n    const viewportWidth = window.innerWidth;\n    const viewportHeight = window.innerHeight;\n\n    // 检查每个位置的可用空间\n    const spaces: Record<string, number> = {\n      top: hostRect.top,\n      bottom: viewportHeight - (hostRect.bottom),\n      left: hostRect.left,\n      right: viewportWidth - (hostRect.right)\n    };\n\n    // 找到最佳位置\n    this.currentPosition = this.positions.reduce((best, current) =>\n      spaces[current] > spaces[best] ? current : best\n    );\n\n    // 根据位置设置样式类\n    this.positions.forEach(pos => this.renderer.removeClass(this.tooltip, pos));\n    this.renderer.addClass(this.tooltip, this.currentPosition);\n\n    // 根据位置计算坐标\n    let top, left;\n    switch(this.currentPosition) {\n      case 'top':\n        top = hostRect.top - tooltipRect.height - 8;\n        left = hostRect.left + (hostRect.width - tooltipRect.width) / 2;\n        break;\n      case 'bottom':\n        top = hostRect.bottom + 8;\n        left = hostRect.left + (hostRect.width - tooltipRect.width) / 2;\n        break;\n      case 'left':\n        top = hostRect.top + (hostRect.height - tooltipRect.height) / 2;\n        left = hostRect.left - tooltipRect.width - 8;\n        break;\n      case 'right':\n        top = hostRect.top + (hostRect.height - tooltipRect.height) / 2;\n        left = hostRect.right + 8;\n        break;\n    }\n\n    // 确保tooltip不超出视口\n    top = Math.max(8, Math.min(viewportHeight - tooltipRect.height - 8, top));\n    left = Math.max(8, Math.min(viewportWidth - tooltipRect.width - 8, left));\n\n    this.renderer.setStyle(this.tooltip, 'top', `${top}px`);\n    this.renderer.setStyle(this.tooltip, 'left', `${left}px`);\n  }\n\n  private hide() {\n    if (this.tooltip) {\n      this.renderer.removeClass(this.tooltip, 'visible');\n      setTimeout(() => {\n        this.viewContainer.clear()\n        this.tooltip = null!\n      }, 300); // 增加动画时间\n    }\n  }\n\n  private clearTimers() {\n    clearTimeout(this.showTimeout);\n    clearTimeout(this.hideTimeout);\n  }\n\n  ngOnDestroy() {\n    this.viewContainer.clear();\n  }\n}\n@Component({\n  selector: 'ngx-file-tooltip',\n  template: `{{ content }}`,\n  standalone: true,\n  styles: [`\n    :host {\n      position: absolute;\n      background: rgba(0, 0, 0, 0.85);\n      color: #fff;\n      font-size: 14px;\n      padding: 6px 8px;\n      border-radius: 2px;\n      box-shadow: 0 3px 6px -4px rgba(0, 0, 0, 0.12), 0 6px 16px 0 rgba(0, 0, 0, 0.08), 0 9px 28px 8px rgba(0, 0, 0, 0.05);\n      max-width: 250px;\n      min-height: 24px;\n      word-wrap: break-word;\n      z-index: 999;\n      pointer-events: none;\n      opacity: 0;\n      display: flex;\n      justify-content: center;\n      align-items: center;\n      transform: scale(0.8);\n      transform-origin: center;\n      transition: opacity 0.2s ease-in-out, transform 0.2s ease-in-out;\n    }\n\n    :host.visible {\n      opacity: 1;\n      transform: scale(1);\n    }\n\n    :host::after {\n      content: '';\n      position: absolute;\n      width: 0;\n      height: 0;\n      border: 5px solid transparent;\n    }\n\n    :host.top::after {\n      border-top-color: rgba(0, 0, 0, 0.85);\n      bottom: -10px;\n      left: 50%;\n      transform: translateX(-50%);\n    }\n\n    :host.bottom::after {\n      border-bottom-color: rgba(0, 0, 0, 0.85);\n      top: -10px;\n      left: 50%;\n      transform: translateX(-50%);\n    }\n\n    :host.left::after {\n      border-left-color: rgba(0, 0, 0, 0.85);\n      right: -10px;\n      top: 50%;\n      transform: translateY(-50%);\n    }\n\n    :host.right::after {\n      border-right-color: rgba(0, 0, 0, 0.85);\n      left: -10px;\n      top: 50%;\n      transform: translateY(-50%);\n    }\n  `]\n})\nexport class TooltipComponent {\n  @Input() content: string = \"\";\n}\n"]}