欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript實現(xiàn)水印效果的示例代碼

 更新時間:2023年05月18日 10:33:40   作者:一個爬坑的Coder  
這篇文章主要為大家詳細介紹了JavaScript如何利用canvas實現(xiàn)添加水印的效果,文中的示例代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學習一下

效果

實現(xiàn)思路

  • 利用canvas繪制出文字
  • 將canvas作為遮罩層背景圖, 將背景x軸和y軸重復

實現(xiàn)步驟

動態(tài)生成canvas并畫出文字

const canvas = document.createElement("canvas");
canvas.width = len * fontSize; // canvas寬度, 目前是根據(jù)文字長度和大小來調(diào)整的, 自己可依照具體需求變動
canvas.height = height + fontSize * 2.8; // canvas高度, 依據(jù)需求調(diào)整

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來對遮罩層做監(jiān)聽, 防止屬性修改或者dom節(jié)點被人為的刪除

MDN: MutationObserver

/**
 * 監(jiān)聽dom變化, 防止水印被篡改
 */
static observeDomChange = (waterMarkDom, options) => {
  const callback = (mutationsList, observer) => {
    for (const mutation of mutationsList) {
      /**
       * 水印節(jié)點的屬性發(fā)生了變動
       */
      if (mutation.target === waterMarkDom) {
        this.setWaterMark(); // 重新生成水印
        observer.disconnect(); // 停止觀察
      }

      /**
       * 強行手動刪除了水印節(jié)點
       */
      if (mutation.removedNodes.length && mutation.removedNodes[0] === waterMarkDom) {
        this.setWaterMark(this.options); // 重新生成水印
        observer.disconnect(); // 停止觀察
      }
    }
  };

  this.observer = new MutationObserver(callback);

  /** 監(jiān)聽body */
  this.observer.observe(document.querySelector("body"), {
    attributes: true, // 觀察屬性變動
    childList: true, // 觀察目標子節(jié)點的變化,是否有添加或者刪除
    subtree: true, // 觀察后代節(jié)點,默認為 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) {
      // 每次重新繪制之前, 需要判斷是否已經(jīng)存在, 如果存在了就先刪除, 再來重新繪制
      document.body.removeChild(waterDom);
    }

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

    const {
      text,
      width, // 寬度是根據(jù)提供的文字大小和文字長度計算出來的, 這里就用不上了
      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);

    /**
     * 監(jiān)聽水印的dom變化
     */
    this.observeDomChange(div);
  };

  /**
   * 去除水印
   */

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

  /**
   * 監(jiān)聽dom變化, 防止水印被篡改
   */
  static observeDomChange = (waterMarkDom, options) => {
    const callback = (mutationsList, observer) => {
      for (const mutation of mutationsList) {
        /**
         * 水印節(jié)點的屬性發(fā)生了變動
         */
        if (mutation.target === waterMarkDom) {
          this.setWaterMark(); // 重新生成水印
          observer.disconnect(); // 停止觀察
        }

        /**
         * 強行手動刪除了水印節(jié)點
         */
        if (mutation.removedNodes.length && mutation.removedNodes[0] === waterMarkDom) {
          this.setWaterMark(this.options); // 重新生成水印
          observer.disconnect();
        }
      }
    };

    this.observer = new MutationObserver(callback);

    /** 監(jiān)聽body */
    this.observer.observe(document.querySelector("body"), {
      attributes: true, // 觀察屬性變動
      childList: true, // 觀察目標子節(jié)點的變化,是否有添加或者刪除
      subtree: true, // 觀察后代節(jié)點,默認為 false
    });
  };
}

Watermark.setWaterMark();

到此這篇關于JavaScript實現(xiàn)水印效果的示例代碼的文章就介紹到這了,更多相關JavaScript水印內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

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

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

    本文為大家介紹下js data日期初始化的常用5種方法,感興趣的朋友可以參考下
    2013-12-12
  • 淺談Javascript Base64 加密解密

    淺談Javascript Base64 加密解密

    這篇文章主要簡單介紹了Javascript Base64 加密解密的使用方法,有需要的小伙伴參考下
    2014-12-12
  • JS實現(xiàn)移動端觸屏拖拽功能

    JS實現(xiàn)移動端觸屏拖拽功能

    這篇文章主要介紹了JS實現(xiàn)移動端觸屏拖拽功能,代碼簡單易懂,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2018-07-07
  • 初學js插入節(jié)點appendChild insertBefore使用方法

    初學js插入節(jié)點appendChild insertBefore使用方法

    由于可見insertBefore()方法的特性是在已有的子節(jié)點前面插入新的節(jié)點但是兩種情況結合起來發(fā)現(xiàn)insertBefore()方法插入節(jié)點,是可以在子節(jié)點列表的任意位置。
    2011-07-07
  • JavaScript函數(shù)中的this四種綁定形式

    JavaScript函數(shù)中的this四種綁定形式

    javascript中的this和函數(shù)息息相關,所以今天,我就給大家詳細地講述一番:javascript函數(shù)中的this ,需要的朋友可以參考下
    2017-08-08
  • jsp 自動編譯機制詳細介紹

    jsp 自動編譯機制詳細介紹

    這篇文章主要介紹了 Jasper的自動檢測實現(xiàn)的機制比較簡單,依靠某后臺線程不斷檢測JSP文件與編譯后的class文件的最后修改時間是否相同,若相同則認為沒有改動,但倘若不同則需要重新編譯,需要的朋友可以參考下
    2016-12-12
  • uniapp小程序開發(fā)組件封裝之自定義輪播圖效果

    uniapp小程序開發(fā)組件封裝之自定義輪播圖效果

    這篇文章主要介紹了uniapp小程序開發(fā)組件封裝之自定義輪播圖,本文主要展示小程序端封裝輪播圖組件,使用的是uniapp進行的開發(fā),主要使用的是uniapp官網(wǎng)提供的swiper組件,需要的朋友可以參考下
    2023-02-02
  • js中this的指向問題歸納總結

    js中this的指向問題歸納總結

    最近發(fā)現(xiàn)在對JS的學習中有很多朋友對this的指向問題還是有很大的誤區(qū)或者說只是大致了解,但是一旦遇到復雜的情況就會因為this指向問題而引發(fā)各種bug。所以這篇文章主要給大家介紹了關于js中this的指向問題的相關資料,需要的朋友可以參考下
    2018-11-11
  • javascript 判斷用戶有沒有操作頁面

    javascript 判斷用戶有沒有操作頁面

    這篇文章主要介紹了javascript 判斷用戶有沒有操作頁面的相關資料,希望通過本文能幫助到大家,需要的朋友可以參考下
    2017-10-10
  • JavaScript 隨機驗證碼的生成實例代碼

    JavaScript 隨機驗證碼的生成實例代碼

    這篇文章主要介紹了JavaScript 隨機驗證碼的生成實例代碼的相關資料,需要的朋友可以參考下
    2016-09-09

最新評論