自適應且不可刪除的水印蒙層

来源:https://www.cnblogs.com/wwwxxxyyy/archive/2022/12/08/16963848.html
-Advertisement-
Play Games

##canvas自適應文字長度,旋轉角度生成水印背景圖 設置canvas字體大小後,通過ctx.measureText(text).width獲取兩行文字的寬度text1,text2,取最大寬度為文本框寬度textWidth 設置兩行文字間距,可得文本框高度:textHeight=2*fontsiz ...


canvas自適應文字長度,旋轉角度生成水印背景圖

  • 設置canvas字體大小後,通過ctx.measureText(text).width獲取兩行文字的寬度text1,text2,取最大寬度為文本框寬度textWidth
  • 設置兩行文字間距,可得文本框高度:textHeight=2*fontsize+ space_line
  • 計算最小一個能夠完全包裹旋轉後文本的盒子寬高
    已知旋轉角度為rotate=>得到弧度rad = (rotate*Math.pi) /180
    單個水印圖平鋪成為蒙層的背景圖,space_x,space_y用於調整水印之間的間距
  function  drawWatermark(el, config = {}) {
      if (!el) return;
      // 預設配置
      let {
        text1 = '今天也要保持愉悅鴨~', //文本1
        text2 = '2022-12-07', // 文本2
        space_x = 0, // x軸間距 
        space_y = 0, // y軸間距
        space_line = 20, //兩列文字的間距
        font = 'Microsoft JhengHei bold',
        fontSize = 40, // 字體
        color = 'rgba(22,22,22,1)', // 字色
        rotate = 30 // 傾斜度
      } = config;
      const canvas =  document.createElement('canvas');
      el.appendChild(canvas);
      const ctx = canvas .getContext('2d');
      ctx.font = fontSize + 'px ' + font; //設置好fontsize才能正確計算出文本寬度
      let tw1= ctx.measureText(text1).width;
      let tw2= ctx.measureText(text2).width;
      let textWidth = Math.max(tw1, tw2); //文本最長寬度為文本框寬度
      let textHeight = fontSize * 2 + space_line; //文本框高度為兩個文本+行間距
      let rad  = (rotate * Math.PI) / 180; //角度轉弧度
      let sin = Math.sin(rad ); 
      let cos = Math.cos(rad );
      let width = textWidth * cos + textHeight * sin + space_x ; //為包裹住文本框的最小盒子寬度
      let height = textWidth * sin + textHeight * cos + space_y; //為包裹住文本框的最小盒子高度
      canvas.width = width;
      canvas.height = height;
      canvas.style.cssText = `width:${width}px;height:${height}px;display:none;`;
      ctx.translate(space_x , textWidth * sin + space_y );  // 移動旋轉中心
      ctx.rotate((-1 * (rotate * Math.PI)) / 180); //旋轉文本框
      ctx.fillStyle = color;
      ctx.textAlign = 'left';
      ctx.textBaseline = 'middle';
      ctx.font = fontSize + 'px ' + font;
      ctx.fillText(text1, (textWidth - tw1) / 2, 0.5 * fontSize);  //文本在文本框中居中顯示
      ctx.fillText(text2, (textWidth - tw2) / 2, 1.5 * fontSize + space_line); //文本在文本框中居中顯示
      return canvas.toDataURL('image/png');
    },

生成蒙層

在目標元素下添加一個相對定位的子元素,將水印圖片平鋪作為背景圖。

禁止蒙層的刪除和修改

  • 刪除或移動element
  • 修改style

transform: translate(100%,100%);
display: none;
visibility: hidden;
取消背景圖

 function createMask(el) {
      //創建蒙層
      let $mask = document.createElement('div');
      //判斷蒙層父元素是否有定位
      let position = window.getComputedStyle(el, null).position;
      if (position === 'static') {
        el.style.position = 'relative';
      }
      //設置蒙層樣式
      let style = `visibility: visible !important;
        transform: translate(0,0)  !important;
        display: block !important;
        visibility: visible !important;
        width: 100% !important; 
        height: 100% !important;
        pointer-events: none !important; 
        background-color: rgba(0, 0, 0, 0)!important;
        background-repeat: repeat !important;
        position: absolute !important;
        top: 0px !important;
        left: 0px !important;
        z-index: 999 !important; 
        background-image: url(${drawWatermark(el)}) !important`;
      $mask.setAttribute('style', style);
      //添加蒙層
      el.append($mask);
      // 創建MutationObserver
      el.observer = new MutationObserver((mutationRecord) => {
        //處理DOM
        mutationRecord.forEach((mutation) => {
          // 蒙層刪除或者被移動到別處
          if (mutation.target === el && mutation.removedNodes[0] == $mask) {
            el.append($mask);
          } else if (mutation.target == $mask && mutation.attributeName === 'style') {
            // 蒙層被更改樣式 在監聽到蒙層樣式更改後,賦值的新的樣式會導致再次觸發監聽回調,所以需要在監聽事件中判斷何時需要賦值
            const changestyle = $mask?.getAttribute('style');
            if (changestyle !== style) {
              $mask.setAttribute('style', style);
            }
          }
        });
      });
      // 啟動監控
      el.observer.observe(el, {
        childList: true,
        attributes: true,
        subtree: true
      });
      return $mask;
    },

行內樣式加important是為了防止通過添加class或其他css覆蓋樣式(暫時沒有找到怎麼如何通過修改css的方式更改樣式的監聽方式)
踩得一個坑
設置元素的行內樣式有很多種

let style = 'width:100%;height:100%' 適用單個樣式更改
element.style.width =100% ;element.style['height'] =100%
element.style.cssText = style
element.setAttribute('style',style )

方式二設置樣式後,行內樣式格式和賦值時的style的格式不一樣,獲取到行內style後直接進行===判斷,回造成死迴圈

解決方法:

  • 第一次監聽到蒙層更改時,立刻移除蒙層,重新生成新蒙層
  • 寫個函數判斷不同格式的兩個樣式屬性上是否相等

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 全網最全的linux上docker安裝oracle的詳細文檔,遇到了n個問題,查了幾十篇文章,最終彙總版,再有解決不了的,私聊我,我幫你解決 1. 拉取阿裡鏡像oracle docker pull registry.cn-hangzhou.aliyuncs.com/helowin/oracle_11 ...
  • 2020年1月,時間跨度長達14年的,微軟2.5億條客戶服務和支持記錄在網上泄露; 同年4月,微盟發生史上最貴“刪庫跑路”事件,造成微盟市值一夜之間縮水約24億港幣; 今年7月,網信辦依據《數據安全法》等法律法規,對滴滴公司開出人民幣80.26億元的巨額罰款,對互聯網企業敲響數據安全警鐘。 數據作為 ...
  • 本文分享自華為雲社區《GaussDB(DWS)字元串、二進位、十六進位互轉》,作者:你是猴子請來的救兵嗎 。 概述 現網中遇到很多小伙伴不清楚字元串與進位之間的轉換方法,其實在GaussDB(DWS)中,進位轉換是非常方便的。這次就來對不同的場景一一進行解析,整理出來供大家翻閱參考。 字元串&二進位 ...
  • 作者:謝澤華 背景 眾所周知單個機房在出現不可抗拒的問題(如斷電、斷網等因素)時,會導致無法正常提供服務,會對業務造成潛在的損失。所以在協同辦公領域,一種可以基於同城或異地多活機制的高可用設計,在保障數據一致性的同時,能夠最大程度降低由於機房的僅單點可用所導致的潛在高可用問題,最大程度上保障業務的用 ...
  • CDC CDC 是 Change Data Capture(變更數據獲取)的簡稱。核心思想是,監測並捕獲資料庫的變動(包括數據或數據表的插入、更新以及刪除等),將這些變更按發生的順序完整記錄下來,寫入到消息中間件中以供其他服務進行訂閱及消費。 CDC 的種類 CDC 主要分為基於查詢和基於 Binl ...
  • 分析服務 ◆ 游戲行業新增“區服分析”埋點模板及分析報告,支持開發者分伺服器查看用戶付費、留存等指標,可進一步評估不同伺服器的玩家質量; ◆ 新增營銷活動報告,可查看廣告任務帶來的曝光、點擊相關信息,讓營銷推廣活動的前端效果一目瞭然; ◆ 新增Web歸因及會話級歸因,以及帶來用戶流量後的行為分析,滿 ...
  • 前幾天在B站刷到尼爾後突發奇想,就想給尼爾做一個簡單的小網站,在思考如何體現尼爾的世界觀的時候想到了使用時間線的方式,將所有時間的事件羅列起來。所以就試著做了一下,這種方式可以很直觀的表現一些歷史上發生的事情,歷史相關主題的一些網站應該可以參考一下 首先來看效果 以上都是游戲里的一些歷史,簡單的設計 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 本篇文章主要總結了微信小程式開發,獲取用戶信息的整個流程步驟。補充了網上很多碎片化的代碼,本人梳理了思路寫下了這篇文章。 思路 1、在js文件中,設置userinfo、hasUserinfo、canIUseGetUserProfile數據 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...