@acrodata/watermark
Version:
Add watermark to your page
177 lines • 25 kB
JavaScript
/** 用于标记是否需要保护 */
export const attributeNameTag = 'data-watermark-tag';
export const observeOptions = {
childList: true,
subtree: true,
attributeFilter: ['style', 'class', attributeNameTag],
};
/** 获取 DataSetKey */
export function getDataSetKey(attributeName) {
return attributeName
.split('-')
.slice(1)
.reduce((prev, cur, index) => {
if (index === 0) {
return cur;
}
return `${prev}${cur[0].toUpperCase() + cur.slice(1)}`;
});
}
/** 将样式对象转换为字符串 */
export const getStyleStr = (style) => {
let str = '';
Object.keys(style).forEach(key => {
const k = key.replace(/([A-Z])/g, '-$1').toLowerCase();
if (style[key] !== '' && style[key] != null) {
str += `${k}:${style[key]};`;
}
});
return str;
};
/** 创建随机 ID */
export const getRandomId = (prefix = '') => {
const uid = window.btoa(decodeURI(encodeURIComponent(prefix)));
return `${uid}-${new Date().getTime()}-${Math.floor(Math.random() * Math.pow(10, 8))}`;
};
/** 获取水印挂载节点 */
export const getContainer = (container) => {
let dom;
if (typeof container === 'string') {
dom = document.querySelector(container);
if (!dom) {
throw new Error(`The watermark container element '${container}' not found!`);
}
}
else {
dom = container ?? document.body;
}
return dom;
};
/** 盲水印解密 */
export const decrypt = (ctx) => {
const originalData = ctx.getImageData(0, 0, ctx.canvas.width, ctx.canvas.height);
const data = originalData.data;
for (let i = 0; i < data.length; i++) {
// 筛选每个像素点的R值
if (i % 4 == 0) {
if (data[i] % 2 == 0) {
// 如果 R 值为偶数,说明这个点是没有水印信息的,将其 R 值设为0
data[i] = 0;
}
else {
// 如果 R 值为奇数,说明这个点是有水印信息的,将其 R 值设为255
data[i] = 255;
}
}
else if (i % 4 == 3) {
// 透明度不作处理
continue;
}
else {
// G、B 值设置为 0,不影响
data[i] = 0;
}
}
// 至此,带有水印信息的点都将展示为 `255,0,0`,而没有水印信息的点将展示为 `0,0,0`,将结果绘制到画布
ctx.putImageData(originalData, 0, 0);
};
export const createHost = (watermarkTag) => {
const dom = document.createElement('div');
// 可以隐藏元素的 CSS 属性
const hiddenCSS = {
'display': 'block !important',
'position': 'static !important',
'opacity': '1 !important',
'visibility': 'visible !important',
'transform': 'none !important',
'clip-path': 'none !important',
};
dom.setAttribute('style', getStyleStr(hiddenCSS));
dom.setAttribute(attributeNameTag, watermarkTag);
return dom;
};
export function getDrawPattern(opts) {
const { text, gapX, gapY, offsetY, offsetX, width, height, rotate, opacity, fontSize, fontStyle, fontVariant, fontWeight, fontFamily, fontColor, textAlign, textBaseline, image, blindText, blindFontSize, blindOpacity, } = opts;
return new Promise((resolve, reject) => {
const canvas = document.createElement('canvas');
const ctx = canvas.getContext('2d');
const ratio = 1;
const canvasWidth = (Number(gapX) + Number(width)) * ratio;
const canvasHeight = (Number(gapY) + Number(height)) * ratio;
const canvasOffsetLeft = Number(offsetX) || Number(gapX) / 2;
const canvasOffsetTop = Number(offsetY) || Number(gapY) / 2;
canvas.setAttribute('width', `${canvasWidth}px`);
canvas.setAttribute('height', `${canvasHeight}px`);
if (ctx) {
const markWidth = width * ratio;
const markHeight = height * ratio;
// 1. 根据元素中心点旋转
ctx.translate(canvasOffsetLeft * ratio, canvasOffsetTop * ratio);
ctx.translate(markWidth / 2, markHeight / 2); // 1
ctx.rotate((Math.PI / 180) * Number(rotate));
ctx.translate(-markWidth / 2, -markHeight / 2); // 1
// 是否需要增加盲水印文字
if (blindText) {
// 盲水印需要低透明度
ctx.globalAlpha = blindOpacity;
ctx.font = `${blindFontSize}px normal`;
ctx.fillText(blindText, 0, 0);
}
// 设置透明度
ctx.globalAlpha = opacity;
// 优先使用图片
if (image) {
const img = new Image();
img.crossOrigin = 'anonymous';
img.referrerPolicy = 'no-referrer';
img.src = image;
img.onload = () => {
ctx.drawImage(img, 0, 0, markWidth, markHeight);
resolve({
url: ctx.canvas.toDataURL(),
width: canvasWidth,
height: canvasHeight,
});
};
return;
}
// 获取文本的最大宽度
const texts = typeof text === 'string' ? text.split('\n') : text;
const widths = texts.map(item => ctx.measureText(item).width);
const maxWidth = Math.max(...widths);
const markSize = Number(fontSize) * ratio;
// 设置文本对齐方式
ctx.textAlign = textAlign;
// 设置文本位置
ctx.textBaseline = textBaseline;
// 设置字体颜色
ctx.fillStyle = fontColor;
// 设置字体
ctx.font = getFont(`${markSize}px`);
// 文案宽度大于画板宽度
if (maxWidth > width) {
ctx.font = getFont(`${markSize / 2}px`);
}
// 多行文本的上下间距
const textGap = 4;
// 获取行高
const lineHeight = markSize;
// 计算水印在 y 轴上的初始位置
let initY = (markHeight - (fontSize + 4) * texts.length - textGap * (texts.length - 1)) / 2;
initY = initY < 0 ? 0 : initY;
for (let i = 0; i < texts.length; i++) {
ctx.fillText(texts[i] || '', markWidth / 2, initY + lineHeight * (i + 1) + textGap * i);
}
resolve({
url: ctx.canvas.toDataURL(),
width: canvasWidth,
height: canvasHeight,
});
}
function getFont(fontSize) {
return `${fontStyle} ${fontVariant} ${fontWeight} ${fontSize} ${fontFamily}`;
}
return reject();
});
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../projects/watermark/src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { WatermarkOptions, DrawPatternResult } from './types';\n\n/** 用于标记是否需要保护 */\nexport const attributeNameTag = 'data-watermark-tag';\n\nexport const observeOptions = {\n  childList: true,\n  subtree: true,\n  attributeFilter: ['style', 'class', attributeNameTag],\n};\n\n/** 获取 DataSetKey */\nexport function getDataSetKey(attributeName: string) {\n  return attributeName\n    .split('-')\n    .slice(1)\n    .reduce((prev, cur, index) => {\n      if (index === 0) {\n        return cur;\n      }\n\n      return `${prev}${cur[0].toUpperCase() + cur.slice(1)}`;\n    });\n}\n\n/** 将样式对象转换为字符串 */\nexport const getStyleStr = (style: Record<string, string | number>) => {\n  let str = '';\n\n  Object.keys(style).forEach(key => {\n    const k = key.replace(/([A-Z])/g, '-$1').toLowerCase();\n    if (style[key] !== '' && style[key] != null) {\n      str += `${k}:${style[key]};`;\n    }\n  });\n\n  return str;\n};\n\n/** 创建随机 ID */\nexport const getRandomId = (prefix = '') => {\n  const uid = window.btoa(decodeURI(encodeURIComponent(prefix)));\n  return `${uid}-${new Date().getTime()}-${Math.floor(Math.random() * Math.pow(10, 8))}`;\n};\n\n/** 获取水印挂载节点 */\nexport const getContainer = (container: WatermarkOptions['container']) => {\n  let dom: HTMLElement | null;\n\n  if (typeof container === 'string') {\n    dom = document.querySelector(container);\n    if (!dom) {\n      throw new Error(`The watermark container element '${container}' not found!`);\n    }\n  } else {\n    dom = container ?? document.body;\n  }\n\n  return dom;\n};\n\n/** 盲水印解密 */\nexport const decrypt = (ctx: CanvasRenderingContext2D) => {\n  const originalData = ctx.getImageData(0, 0, ctx.canvas.width, ctx.canvas.height);\n  const data = originalData.data;\n  for (let i = 0; i < data.length; i++) {\n    // 筛选每个像素点的R值\n    if (i % 4 == 0) {\n      if (data[i] % 2 == 0) {\n        // 如果 R 值为偶数，说明这个点是没有水印信息的，将其 R 值设为0\n        data[i] = 0;\n      } else {\n        // 如果 R 值为奇数，说明这个点是有水印信息的，将其 R 值设为255\n        data[i] = 255;\n      }\n    } else if (i % 4 == 3) {\n      // 透明度不作处理\n      continue;\n    } else {\n      // G、B 值设置为 0，不影响\n      data[i] = 0;\n    }\n  }\n  // 至此，带有水印信息的点都将展示为 `255,0,0`，而没有水印信息的点将展示为 `0,0,0`，将结果绘制到画布\n  ctx.putImageData(originalData, 0, 0);\n};\n\nexport const createHost = (watermarkTag: string) => {\n  const dom = document.createElement('div');\n  // 可以隐藏元素的 CSS 属性\n  const hiddenCSS = {\n    'display': 'block !important',\n    'position': 'static !important',\n    'opacity': '1 !important',\n    'visibility': 'visible !important',\n    'transform': 'none !important',\n    'clip-path': 'none !important',\n  };\n  dom.setAttribute('style', getStyleStr(hiddenCSS));\n  dom.setAttribute(attributeNameTag, watermarkTag);\n  return dom;\n};\n\nexport function getDrawPattern(opts: WatermarkOptions): Promise<DrawPatternResult> {\n  const {\n    text,\n    gapX,\n    gapY,\n    offsetY,\n    offsetX,\n    width,\n    height,\n    rotate,\n    opacity,\n    fontSize,\n    fontStyle,\n    fontVariant,\n    fontWeight,\n    fontFamily,\n    fontColor,\n    textAlign,\n    textBaseline,\n    image,\n    blindText,\n    blindFontSize,\n    blindOpacity,\n  } = opts as Required<WatermarkOptions>;\n  return new Promise((resolve, reject) => {\n    const canvas = document.createElement('canvas');\n\n    const ctx = canvas.getContext('2d');\n    const ratio = 1;\n\n    const canvasWidth = (Number(gapX) + Number(width)) * ratio;\n    const canvasHeight = (Number(gapY) + Number(height)) * ratio;\n    const canvasOffsetLeft = Number(offsetX) || Number(gapX) / 2;\n    const canvasOffsetTop = Number(offsetY) || Number(gapY) / 2;\n\n    canvas.setAttribute('width', `${canvasWidth}px`);\n    canvas.setAttribute('height', `${canvasHeight}px`);\n\n    if (ctx) {\n      const markWidth = width * ratio;\n      const markHeight = height * ratio;\n\n      // 1. 根据元素中心点旋转\n      ctx.translate(canvasOffsetLeft * ratio, canvasOffsetTop * ratio);\n      ctx.translate(markWidth / 2, markHeight / 2); // 1\n      ctx.rotate((Math.PI / 180) * Number(rotate));\n      ctx.translate(-markWidth / 2, -markHeight / 2); // 1\n\n      // 是否需要增加盲水印文字\n      if (blindText) {\n        // 盲水印需要低透明度\n        ctx.globalAlpha = blindOpacity;\n        ctx.font = `${blindFontSize}px normal`;\n        ctx.fillText(blindText, 0, 0);\n      }\n\n      // 设置透明度\n      ctx.globalAlpha = opacity;\n\n      // 优先使用图片\n      if (image) {\n        const img = new Image();\n        img.crossOrigin = 'anonymous';\n        img.referrerPolicy = 'no-referrer';\n        img.src = image;\n        img.onload = () => {\n          ctx.drawImage(img, 0, 0, markWidth, markHeight);\n          resolve({\n            url: ctx.canvas.toDataURL(),\n            width: canvasWidth,\n            height: canvasHeight,\n          });\n        };\n        return;\n      }\n\n      // 获取文本的最大宽度\n      const texts = typeof text === 'string' ? text.split('\\n') : text;\n      const widths = texts.map(item => ctx.measureText(item).width);\n      const maxWidth = Math.max(...widths);\n\n      const markSize = Number(fontSize) * ratio;\n\n      // 设置文本对齐方式\n      ctx.textAlign = textAlign;\n      // 设置文本位置\n      ctx.textBaseline = textBaseline;\n      // 设置字体颜色\n      ctx.fillStyle = fontColor;\n      // 设置字体\n      ctx.font = getFont(`${markSize}px`);\n\n      // 文案宽度大于画板宽度\n      if (maxWidth > width) {\n        ctx.font = getFont(`${markSize / 2}px`);\n      }\n\n      // 多行文本的上下间距\n      const textGap = 4;\n\n      // 获取行高\n      const lineHeight = markSize;\n\n      // 计算水印在 y 轴上的初始位置\n      let initY = (markHeight - (fontSize + 4) * texts.length - textGap * (texts.length - 1)) / 2;\n      initY = initY < 0 ? 0 : initY;\n\n      for (let i = 0; i < texts.length; i++) {\n        ctx.fillText(texts[i] || '', markWidth / 2, initY + lineHeight * (i + 1) + textGap * i);\n      }\n      resolve({\n        url: ctx.canvas.toDataURL(),\n        width: canvasWidth,\n        height: canvasHeight,\n      });\n    }\n\n    function getFont(fontSize: string) {\n      return `${fontStyle} ${fontVariant} ${fontWeight} ${fontSize} ${fontFamily}`;\n    }\n\n    return reject();\n  });\n}\n"]}