UNPKG

@acrodata/watermark

Version:
185 lines 22.7 kB
import { attributeNameTag, createHost, decrypt, getContainer, getDataSetKey, getDrawPattern, getRandomId, getStyleStr, observeOptions, } from './utils'; export const defaultOptions = { gapX: 100, gapY: 100, offsetX: 0, offsetY: 0, width: 120, height: 60, opacity: 0.15, rotate: -24, fontSize: 16, fontWeight: '400', fontStyle: 'normal', fontVariant: 'normal', fontColor: '#000', fontFamily: 'sans-serif', textAlign: 'center', textBaseline: 'alphabetic', secure: true, blindFontSize: 16, blindOpacity: 0.005, repeat: 'multiply', zIndex: 9999, }; export class Watermark { /** 水印配置 */ options = {}; /** 水印挂载容器 */ container; /** 水印的宿主节点 */ watermarkHost; /** 水印节点 */ watermarkDom; /** 水印样式 */ style = { pointerEvents: 'none', position: 'absolute', inset: 0, }; watermarkTag = getRandomId('watermark'); shadowRoot; mutationObserver = null; constructor(options = {}) { this.options = Object.assign({}, defaultOptions, options); this._render(); } static decrypt = decrypt; update(options = {}) { this.options = { ...this.options, ...options, }; this._render(); } show() { if (this.watermarkDom) { this.style['display'] = 'block'; this.watermarkDom.setAttribute('style', getStyleStr(this.style)); } } hide() { if (this.watermarkDom) { this.style['display'] = 'none'; this.watermarkDom.setAttribute('style', getStyleStr(this.style)); } } destroy() { this.shadowRoot = undefined; if (this.watermarkHost) { this.watermarkHost.remove(); this.watermarkHost = undefined; } if (this.watermarkDom) { this.watermarkDom.remove(); this.watermarkDom = undefined; } this._destroyMutationObserver(); } _shouldRerender = (mutation) => { // 修改样式或属性 if (mutation.type === 'attributes') { if (mutation.attributeName === attributeNameTag) { return true; } if (this.watermarkTag === this._getNodeRandomId(mutation.target)) { return true; } } // 删除节点 if (mutation.removedNodes.length && this.watermarkTag === this._getNodeRandomId(mutation.removedNodes[0])) { return true; } return false; }; _getNodeRandomId = (node) => { return node?.dataset?.[getDataSetKey(attributeNameTag)]; }; _destroyMutationObserver = () => { if (this.mutationObserver) { this.mutationObserver.takeRecords(); this.mutationObserver.disconnect(); this.mutationObserver = null; } }; _getWatermarkDom = async () => { if (!this.watermarkDom) { this.watermarkDom = document.createElement('div'); } const bgConfig = await getDrawPattern(this.options); if (bgConfig?.url) { const bgImg = bgConfig.url; this.style['zIndex'] = this.options.zIndex; if (this.options.repeat === 'multiply') { this.style['backgroundImage'] = `url(${bgImg}), url(${bgImg})`; this.style['backgroundRepeat'] = 'repeat'; this.style['backgroundPosition'] = `${bgConfig.width / 2}px ${bgConfig.height / 2}px, 0 0`; } else { this.style['backgroundImage'] = `url(${bgImg})`; this.style['backgroundRepeat'] = 'repeat'; this.style['backgroundPosition'] = ''; if (this.options.repeat === 'none') { this.style['backgroundRepeat'] = 'no-repeat'; this.style['backgroundPosition'] = this.options.position || 'center'; } } if (!this.options.container) { this.style['position'] = 'fixed'; } if (this.options.scrollHeight) { const height = this.options.scrollHeight; this.style['height'] = isNaN(Number(height)) ? height : height + 'px'; } this.watermarkDom.setAttribute('style', getStyleStr(this.style)); } this.watermarkDom.setAttribute(attributeNameTag, this.watermarkTag); return this.watermarkDom; }; async _render() { this._destroyMutationObserver(); // 获取水印挂载节点 this.container = getContainer(this.options.container); // 获取水印父节点 if (!this.watermarkHost) { this.watermarkHost = createHost(this.watermarkTag); this.container.append(this.watermarkHost); } // 获取水印 DOM this.watermarkDom = await this._getWatermarkDom(); // 删除已有水印 if (this.watermarkHost) { const children = this.watermarkHost.childNodes || []; children.forEach(child => { this.watermarkHost.removeChild(child); }); } // 判断是否支持 Shadow DOM if (typeof this.watermarkHost.attachShadow === 'function') { if (!this.shadowRoot) { this.shadowRoot = this.watermarkHost.attachShadow({ mode: 'open' }); } } else { this.shadowRoot = this.watermarkHost; } this.shadowRoot.append(this.watermarkDom); if (MutationObserver && this.options.secure) { this.mutationObserver = new MutationObserver(mutations => { mutations.forEach(mutation => { if (this._shouldRerender(mutation)) { this.destroy(); this._render(); return; } }); }); this.mutationObserver.observe(this.container, observeOptions); if (this.shadowRoot) { this.mutationObserver.observe(this.shadowRoot, observeOptions); } } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watermark.js","sourceRoot":"","sources":["../../../../projects/watermark/src/lib/watermark.ts"],"names":[],"mappings":"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","sourcesContent":["import { WatermarkOptions } from './types';\nimport {\n  attributeNameTag,\n  createHost,\n  decrypt,\n  getContainer,\n  getDataSetKey,\n  getDrawPattern,\n  getRandomId,\n  getStyleStr,\n  observeOptions,\n} from './utils';\n\nexport const defaultOptions: WatermarkOptions = {\n  gapX: 100,\n  gapY: 100,\n  offsetX: 0,\n  offsetY: 0,\n  width: 120,\n  height: 60,\n  opacity: 0.15,\n  rotate: -24,\n  fontSize: 16,\n  fontWeight: '400',\n  fontStyle: 'normal',\n  fontVariant: 'normal',\n  fontColor: '#000',\n  fontFamily: 'sans-serif',\n  textAlign: 'center',\n  textBaseline: 'alphabetic',\n  secure: true,\n  blindFontSize: 16,\n  blindOpacity: 0.005,\n  repeat: 'multiply',\n  zIndex: 9999,\n};\n\nexport class Watermark {\n  /** 水印配置 */\n  private options: WatermarkOptions = {};\n  /** 水印挂载容器 */\n  private container?: HTMLElement;\n  /** 水印的宿主节点 */\n  private watermarkHost?: HTMLElement;\n  /** 水印节点 */\n  private watermarkDom?: HTMLElement;\n  /** 水印样式 */\n  private style: Record<string, string | number> = {\n    pointerEvents: 'none',\n    position: 'absolute',\n    inset: 0,\n  };\n  private watermarkTag = getRandomId('watermark');\n  private shadowRoot?: ShadowRoot | HTMLElement;\n  private mutationObserver: MutationObserver | null = null;\n\n  constructor(options: WatermarkOptions = {}) {\n    this.options = Object.assign({}, defaultOptions, options);\n\n    this._render();\n  }\n\n  static decrypt = decrypt;\n\n  update(options: WatermarkOptions = {}) {\n    this.options = {\n      ...this.options,\n      ...options,\n    };\n\n    this._render();\n  }\n\n  show() {\n    if (this.watermarkDom) {\n      this.style['display'] = 'block';\n      this.watermarkDom.setAttribute('style', getStyleStr(this.style));\n    }\n  }\n\n  hide() {\n    if (this.watermarkDom) {\n      this.style['display'] = 'none';\n      this.watermarkDom.setAttribute('style', getStyleStr(this.style));\n    }\n  }\n\n  destroy() {\n    this.shadowRoot = undefined;\n\n    if (this.watermarkHost) {\n      this.watermarkHost.remove();\n      this.watermarkHost = undefined;\n    }\n\n    if (this.watermarkDom) {\n      this.watermarkDom.remove();\n      this.watermarkDom = undefined;\n    }\n\n    this._destroyMutationObserver();\n  }\n\n  _shouldRerender = (mutation: MutationRecord) => {\n    // 修改样式或属性\n    if (mutation.type === 'attributes') {\n      if (mutation.attributeName === attributeNameTag) {\n        return true;\n      }\n      if (this.watermarkTag === this._getNodeRandomId(mutation.target)) {\n        return true;\n      }\n    }\n\n    // 删除节点\n    if (\n      mutation.removedNodes.length &&\n      this.watermarkTag === this._getNodeRandomId(mutation.removedNodes[0])\n    ) {\n      return true;\n    }\n\n    return false;\n  };\n\n  _getNodeRandomId = (node: Node) => {\n    return (node as HTMLElement)?.dataset?.[getDataSetKey(attributeNameTag)];\n  };\n\n  _destroyMutationObserver = () => {\n    if (this.mutationObserver) {\n      this.mutationObserver.takeRecords();\n      this.mutationObserver.disconnect();\n      this.mutationObserver = null;\n    }\n  };\n\n  _getWatermarkDom = async () => {\n    if (!this.watermarkDom) {\n      this.watermarkDom = document.createElement('div');\n    }\n\n    const bgConfig = await getDrawPattern(this.options);\n\n    if (bgConfig?.url) {\n      const bgImg = bgConfig.url;\n\n      this.style['zIndex'] = this.options.zIndex!;\n\n      if (this.options.repeat === 'multiply') {\n        this.style['backgroundImage'] = `url(${bgImg}), url(${bgImg})`;\n        this.style['backgroundRepeat'] = 'repeat';\n        this.style['backgroundPosition'] = `${bgConfig.width / 2}px ${bgConfig.height / 2}px, 0 0`;\n      } else {\n        this.style['backgroundImage'] = `url(${bgImg})`;\n        this.style['backgroundRepeat'] = 'repeat';\n        this.style['backgroundPosition'] = '';\n\n        if (this.options.repeat === 'none') {\n          this.style['backgroundRepeat'] = 'no-repeat';\n          this.style['backgroundPosition'] = this.options.position || 'center';\n        }\n      }\n\n      if (!this.options.container) {\n        this.style['position'] = 'fixed';\n      }\n\n      if (this.options.scrollHeight) {\n        const height = this.options.scrollHeight;\n        this.style['height'] = isNaN(Number(height)) ? height : height + 'px';\n      }\n\n      this.watermarkDom.setAttribute('style', getStyleStr(this.style));\n    }\n\n    this.watermarkDom.setAttribute(attributeNameTag, this.watermarkTag);\n\n    return this.watermarkDom;\n  };\n\n  async _render() {\n    this._destroyMutationObserver();\n\n    // 获取水印挂载节点\n    this.container = getContainer(this.options.container);\n    // 获取水印父节点\n    if (!this.watermarkHost) {\n      this.watermarkHost = createHost(this.watermarkTag);\n      this.container.append(this.watermarkHost);\n    }\n\n    // 获取水印 DOM\n    this.watermarkDom = await this._getWatermarkDom();\n\n    // 删除已有水印\n    if (this.watermarkHost) {\n      const children = this.watermarkHost.childNodes || [];\n\n      children.forEach(child => {\n        this.watermarkHost!.removeChild(child);\n      });\n    }\n\n    // 判断是否支持 Shadow DOM\n    if (typeof this.watermarkHost.attachShadow === 'function') {\n      if (!this.shadowRoot) {\n        this.shadowRoot = this.watermarkHost.attachShadow({ mode: 'open' });\n      }\n    } else {\n      this.shadowRoot = this.watermarkHost;\n    }\n\n    this.shadowRoot.append(this.watermarkDom);\n\n    if (MutationObserver && this.options.secure) {\n      this.mutationObserver = new MutationObserver(mutations => {\n        mutations.forEach(mutation => {\n          if (this._shouldRerender(mutation)) {\n            this.destroy();\n            this._render();\n            return;\n          }\n        });\n      });\n      this.mutationObserver.observe(this.container, observeOptions);\n      if (this.shadowRoot) {\n        this.mutationObserver.observe(this.shadowRoot, observeOptions);\n      }\n    }\n  }\n}\n"]}