JavaScript实现水印效果的示例代码

 更新时间:2023年05月18日 10:33:40   作者:一个爬坑的Coder  
这篇文章主要为大家详细介绍了JavaScript如何利用canvas实现添加水印的效果,文中的示例代码简洁易懂,感兴趣的小伙伴可以跟随小编一起学习一下

效果

实现思路

  • 利用canvas绘制出文字
  • 将canvas作为遮罩层背景图, 将背景x轴和y轴重复

实现步骤

动态生成canvas并画出文字

const canvas = document.createElement("canvas");
canvas.width = len * fontSize; // canvas宽度, 目前是根据文字长度和大小来调整的, 自己可依照具体需求变动
canvas.height = height + fontSize * 2.8; // canvas高度, 依据需求调整

const context = canvas.getContext("2d");
context.translate(0, canvas.height / 2); // 改变旋转基点
context.rotate((-rotate * Math.PI) / 180); // 进行旋转, 传过来的旋转角度
context.font = `${fontSize}px Vedana`; // 设置字体
context.fillStyle = color; // 设置文字颜色

// 将需要的文本, 绘制到canvas上面
context.fillText(text, 10, canvas.height / 2 - 100);

将canvas做为遮罩层背景图

// 生成水印遮罩层
const div = document.createElement("div");
div.id = DOM_ID;
div.style.pointerEvents = "none";
div.style.position = "fixed";
div.style.zIndex = zIndex;
div.style.left = "-32%";
div.style.top = "-32%";
div.style.opacity = opacity;
div.style.width = "150%";
div.style.height = "150%";
div.style.background = `url('${canvas.toDataURL("images/png")}')repeat left top`;

document.body.appendChild(div);

防止篡改水印

利用MutationObserverAPI来对遮罩层做监听, 防止属性修改或者dom节点被人为的删除

MDN: MutationObserver

/**
 * 监听dom变化, 防止水印被篡改
 */
static observeDomChange = (waterMarkDom, options) => {
  const callback = (mutationsList, observer) => {
    for (const mutation of mutationsList) {
      /**
       * 水印节点的属性发生了变动
       */
      if (mutation.target === waterMarkDom) {
        this.setWaterMark(); // 重新生成水印
        observer.disconnect(); // 停止观察
      }

      /**
       * 强行手动删除了水印节点
       */
      if (mutation.removedNodes.length && mutation.removedNodes[0] === waterMarkDom) {
        this.setWaterMark(this.options); // 重新生成水印
        observer.disconnect(); // 停止观察
      }
    }
  };

  this.observer = new MutationObserver(callback);

  /** 监听body */
  this.observer.observe(document.querySelector("body"), {
    attributes: true, // 观察属性变动
    childList: true, // 观察目标子节点的变化,是否有添加或者删除
    subtree: true, // 观察后代节点,默认为 false
  });
};

所有代码

const DOM_ID = "yss-cj-create";

/**
 * 水印的默认属性
 */
const DEFAULT_OPTIONS = {
  text: "cxk  管理员  20230424",
  width: 520, // 水印块的宽度
  height: 280, // 水印块的高度
  rotate: 20, // 水印块的旋转角度
  fontSize: 28, // 文字大小
  color: "#666", // 文字颜色
  opacity: "0.3", // 遮罩层的透明度
  zIndex: "9999999999", // 遮罩层的层级
};

class Watermark {
  options = {};
  observer = null;

  /**
   * 生成水印
   */
  static setWaterMark = (options = {}) => {
    const waterDom = document.getElementById(DOM_ID);
    if (waterDom !== null) {
      // 每次重新绘制之前, 需要判断是否已经存在, 如果存在了就先删除, 再来重新绘制
      document.body.removeChild(waterDom);
    }

    const latestOptions = { ...DEFAULT_OPTIONS, ...options };
    this.options = latestOptions;

    const {
      text,
      width, // 宽度是根据提供的文字大小和文字长度计算出来的, 这里就用不上了
      height, // 水印块的高度
      rotate, // 水印块的旋转角度
      fontSize, // 文字大小
      color, // 文字颜色
      opacity, // 遮罩层的透明度
      zIndex, // 遮罩层的层级
    } = latestOptions;

    const len = text.length;
    const canvas = document.createElement("canvas");
    canvas.width = len * fontSize;
    canvas.height = height + fontSize * 2.8;

    const context = canvas.getContext("2d");
    context.translate(0, canvas.height / 2);
    context.rotate((-rotate * Math.PI) / 180);
    context.font = `${fontSize}px Vedana`; // 设置字体
    context.fillStyle = color; // 设置文字颜色

    // 将需要的文本, 绘制到canvas上面
    context.fillText(text, 10, canvas.height / 2 - 100);

    // 生成水印遮罩层
    const div = document.createElement("div");
    div.id = DOM_ID;
    div.style.pointerEvents = "none";
    div.style.position = "fixed";
    div.style.zIndex = zIndex;
    div.style.left = "-32%";
    div.style.top = "-32%";
    div.style.opacity = opacity;
    div.style.width = "150%";
    div.style.height = "150%";
    div.style.background = `url('${canvas.toDataURL("images/png")}')repeat left top`;

    document.body.appendChild(div);

    /**
     * 监听水印的dom变化
     */
    this.observeDomChange(div);
  };

  /**
   * 去除水印
   */

  static removeWatermark = () => {
    const dom = document.getElementById(DOM_ID);
    if (dom !== null) {
      document.body.removeChild(dom);
    }
  };

  /**
   * 监听dom变化, 防止水印被篡改
   */
  static observeDomChange = (waterMarkDom, options) => {
    const callback = (mutationsList, observer) => {
      for (const mutation of mutationsList) {
        /**
         * 水印节点的属性发生了变动
         */
        if (mutation.target === waterMarkDom) {
          this.setWaterMark(); // 重新生成水印
          observer.disconnect(); // 停止观察
        }

        /**
         * 强行手动删除了水印节点
         */
        if (mutation.removedNodes.length && mutation.removedNodes[0] === waterMarkDom) {
          this.setWaterMark(this.options); // 重新生成水印
          observer.disconnect();
        }
      }
    };

    this.observer = new MutationObserver(callback);

    /** 监听body */
    this.observer.observe(document.querySelector("body"), {
      attributes: true, // 观察属性变动
      childList: true, // 观察目标子节点的变化,是否有添加或者删除
      subtree: true, // 观察后代节点,默认为 false
    });
  };
}

Watermark.setWaterMark();

到此这篇关于JavaScript实现水印效果的示例代码的文章就介绍到这了,更多相关JavaScript水印内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js data日期初始化的5种方法

    js data日期初始化的5种方法

    本文为大家介绍下js data日期初始化的常用5种方法,感兴趣的朋友可以参考下
    2013-12-12
  • 浅谈Javascript Base64 加密解密

    浅谈Javascript Base64 加密解密

    这篇文章主要简单介绍了Javascript Base64 加密解密的使用方法,有需要的小伙伴参考下
    2014-12-12
  • JS实现移动端触屏拖拽功能

    JS实现移动端触屏拖拽功能

    这篇文章主要介绍了JS实现移动端触屏拖拽功能,代码简单易懂,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-07-07
  • 初学js插入节点appendChild insertBefore使用方法

    初学js插入节点appendChild insertBefore使用方法

    由于可见insertBefore()方法的特性是在已有的子节点前面插入新的节点但是两种情况结合起来发现insertBefore()方法插入节点,是可以在子节点列表的任意位置。
    2011-07-07
  • JavaScript函数中的this四种绑定形式

    JavaScript函数中的this四种绑定形式

    javascript中的this和函数息息相关,所以今天,我就给大家详细地讲述一番:javascript函数中的this ,需要的朋友可以参考下
    2017-08-08
  • jsp 自动编译机制详细介绍

    jsp 自动编译机制详细介绍

    这篇文章主要介绍了 Jasper的自动检测实现的机制比较简单,依靠某后台线程不断检测JSP文件与编译后的class文件的最后修改时间是否相同,若相同则认为没有改动,但倘若不同则需要重新编译,需要的朋友可以参考下
    2016-12-12
  • uniapp小程序开发组件封装之自定义轮播图效果

    uniapp小程序开发组件封装之自定义轮播图效果

    这篇文章主要介绍了uniapp小程序开发组件封装之自定义轮播图,本文主要展示小程序端封装轮播图组件,使用的是uniapp进行的开发,主要使用的是uniapp官网提供的swiper组件,需要的朋友可以参考下
    2023-02-02
  • js中this的指向问题归纳总结

    js中this的指向问题归纳总结

    最近发现在对JS的学习中有很多朋友对this的指向问题还是有很大的误区或者说只是大致了解,但是一旦遇到复杂的情况就会因为this指向问题而引发各种bug。所以这篇文章主要给大家介绍了关于js中this的指向问题的相关资料,需要的朋友可以参考下
    2018-11-11
  • javascript 判断用户有没有操作页面

    javascript 判断用户有没有操作页面

    这篇文章主要介绍了javascript 判断用户有没有操作页面的相关资料,希望通过本文能帮助到大家,需要的朋友可以参考下
    2017-10-10
  • JavaScript 随机验证码的生成实例代码

    JavaScript 随机验证码的生成实例代码

    这篇文章主要介绍了JavaScript 随机验证码的生成实例代码的相关资料,需要的朋友可以参考下
    2016-09-09

最新评论