UNPKG

@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.

137 lines 15.3 kB
import { ApplicationRef, createComponent, inject, Injectable } from '@angular/core'; import { BehaviorSubject } from 'rxjs'; import { PreviewModalComponent } from '../components'; import { ThemeService } from "./theme.service"; import { I18nUtils } from "../i18n/i18n.utils"; import * as i0 from "@angular/core"; export const INITIAL_PREVIEW_STATE = { isVisible: false, currentIndex: 0, files: [] }; export class PreviewService { constructor() { this.appRef = inject(ApplicationRef); this.lang = 'zh'; this.loading = new BehaviorSubject(false); // endregion // region 状态管理 this.stateSubject = new BehaviorSubject(INITIAL_PREVIEW_STATE); } /** * 初始化 * @param injector * @param envInjector */ init(injector, envInjector) { this.envInjector = envInjector; this.injector = injector; } /** * 设置语言 * @param lang */ setLang(lang) { this.lang = lang; } /** * 获取实际的lang parser */ getLangParser() { return I18nUtils.get(this.lang); } get state() { return this.stateSubject.getValue(); } getStateObservable() { return this.stateSubject.asObservable(); } previous() { const state = this.state; const newIndex = Math.max(0, state.currentIndex - 1); this.updatePreviewState(true, state.files, newIndex); } next() { const state = this.state; const newIndex = Math.min(state.files.length - 1, state.currentIndex + 1); this.updatePreviewState(true, state.files, newIndex); } updatePreviewState(isVisible, files, index) { const currentFile = files[index]; this.stateSubject.next({ isVisible, currentFile, currentIndex: index, files }); } /** * 设置加载中状态 * @param loading */ setLoading(loading) { this.loading.next(loading); } getLoadingObservable() { return this.loading.asObservable(); } get modalElement() { return this.modalRef?.location.nativeElement; } open(options) { const { files, index = 0 } = options; if (this.modalRef) { this.cleanupModal(); } try { this.modalRef = createComponent(PreviewModalComponent, { environmentInjector: this.envInjector, elementInjector: this.injector, }); Object.assign(this.modalRef.instance, options); this.injector.get(ThemeService).bindElement(this.modalRef.location.nativeElement); document.body.appendChild(this.modalRef.location.nativeElement); this.modalRef.changeDetectorRef.detectChanges(); this.updatePreviewState(true, files, index); this.appRef.attachView(this.modalRef.hostView); } catch (error) { console.error('Error creating preview-list modal:', error); this.cleanupModal(); } } close() { if (document.fullscreenElement) { document?.exitFullscreen(); } this.updatePreviewState(false, [], 0); this.cleanupModal(); } cleanupModal() { if (!this.modalRef) return; try { // 从 DOM 中移除模态框 const element = this.modalRef.location.nativeElement; if (element.parentNode) { element.parentNode.removeChild(element); } // 从 ApplicationRef 中分离视图 this.appRef.detachView(this.modalRef.hostView); // 销毁组件 this.modalRef.destroy(); } catch (error) { console.error('Error cleaning up modal:', error); } finally { this.modalRef = undefined; } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PreviewService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PreviewService }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PreviewService, decorators: [{ type: Injectable }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"preview.service.js","sourceRoot":"","sources":["../../../../../libs/ngx-file-preview/src/lib/services/preview.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ApplicationRef,\n  ComponentRef,\n  createComponent,\n  EnvironmentInjector,\n  inject,\n  Injectable,\n  Injector\n} from '@angular/core';\nimport {PreviewFile, PreviewOptions} from '../types/preview.types';\nimport {BehaviorSubject} from 'rxjs';\nimport {PreviewModalComponent} from '../components';\nimport {ThemeService} from \"./theme.service\";\nimport {I18nUtils} from \"../i18n/i18n.utils\";\n\nexport const INITIAL_PREVIEW_STATE = {\n  isVisible: false,\n  currentIndex: 0,\n  files: []\n}\n\nexport interface PreviewState {\n  isVisible: boolean;\n  currentFile?: PreviewFile;\n  currentIndex: number;\n  files: PreviewFile[];\n}\n\n@Injectable()\nexport class PreviewService {\n  // region 服务管理\n  /**\n   * 初始化 需要将所有service注入到modal\n   */\n  private injector!: Injector;\n  private envInjector!: EnvironmentInjector;\n  private appRef = inject(ApplicationRef)\n  private lang: string = 'zh';\n  private loading: BehaviorSubject<boolean> = new BehaviorSubject(false);\n\n  /**\n   * 初始化\n   * @param injector\n   * @param envInjector\n   */\n  init(injector: Injector, envInjector: EnvironmentInjector) {\n    this.envInjector = envInjector;\n    this.injector = injector;\n  }\n\n  /**\n   * 设置语言\n   * @param lang\n   */\n  setLang(lang: string) {\n    this.lang = lang;\n  }\n\n  /**\n   * 获取实际的lang parser\n   */\n  getLangParser() {\n    return I18nUtils.get(this.lang)\n  }\n\n  // endregion\n  // region 状态管理\n  readonly stateSubject = new BehaviorSubject<PreviewState>(INITIAL_PREVIEW_STATE);\n\n  get state() {\n    return this.stateSubject.getValue();\n  }\n\n  getStateObservable() {\n    return this.stateSubject.asObservable();\n  }\n\n  previous() {\n    const state = this.state;\n    const newIndex = Math.max(0, state.currentIndex - 1);\n    this.updatePreviewState(true, state.files, newIndex);\n  }\n\n  next() {\n    const state = this.state;\n    const newIndex = Math.min(state.files.length - 1, state.currentIndex + 1);\n    this.updatePreviewState(true, state.files, newIndex);\n  }\n\n  private updatePreviewState(isVisible: boolean, files: PreviewFile[], index: number) {\n    const currentFile = files[index];\n    this.stateSubject.next({\n      isVisible,\n      currentFile,\n      currentIndex: index,\n      files\n    });\n  }\n\n  /**\n   * 设置加载中状态\n   * @param loading\n   */\n  setLoading(loading: boolean) {\n    this.loading.next(loading);\n  }\n\n  getLoadingObservable() {\n    return this.loading.asObservable();\n  }\n\n  // endregion\n  // region Modal管理\n  private modalRef?: ComponentRef<PreviewModalComponent>;\n\n  get modalElement() {\n    return this.modalRef?.location.nativeElement\n  }\n\n  open(options: PreviewOptions) {\n    const {files, index = 0} = options;\n    if (this.modalRef) {\n      this.cleanupModal()\n    }\n    try {\n      this.modalRef = createComponent(PreviewModalComponent, {\n        environmentInjector: this.envInjector,\n        elementInjector: this.injector,\n      });\n      Object.assign(this.modalRef.instance, options)\n      this.injector.get(ThemeService).bindElement(this.modalRef.location.nativeElement)\n      document.body.appendChild(this.modalRef.location.nativeElement);\n      this.modalRef.changeDetectorRef.detectChanges();\n      this.updatePreviewState(true, files, index);\n      this.appRef.attachView(this.modalRef.hostView);\n    } catch (error) {\n      console.error('Error creating preview-list modal:', error);\n      this.cleanupModal();\n    }\n  }\n\n  close() {\n    if (document.fullscreenElement) {\n      document?.exitFullscreen();\n    }\n    this.updatePreviewState(false, [], 0);\n    this.cleanupModal();\n  }\n\n  private cleanupModal() {\n    if (!this.modalRef) return;\n    try {\n      // 从 DOM 中移除模态框\n      const element = this.modalRef.location.nativeElement;\n      if (element.parentNode) {\n        element.parentNode.removeChild(element);\n      }\n      // 从 ApplicationRef 中分离视图\n      this.appRef.detachView(this.modalRef.hostView);\n      // 销毁组件\n      this.modalRef.destroy();\n    } catch (error) {\n      console.error('Error cleaning up modal:', error);\n    } finally {\n      this.modalRef = undefined;\n    }\n  }\n\n  // endregion\n\n\n}\n"]}