详解如何用JS实现覆盖水印效果

废话开篇:简单实现一个覆盖水印的小功能,水印一般都是添加在图片上,然后直接加载处理过的图片url即可,这里并没有修改图片,而是直接的在待添加水印的 dom 上添加一个 canvas 蒙版。

一、效果

处理之前

DIV

image.png

IMG

image.png

处理之后

DIV

image.png

IMG

image.png

这里添加 “水印”(其实并不是真正的水印) 到 DIV  的时候按钮点击事件并不会因为有蒙版遮挡而无法点击

火龙果写作 火龙果写作

用火龙果,轻松写作,通过校对、改写、扩展等功能实现高质量内容生产。

火龙果写作 106 查看详情 火龙果写作

二、JS 代码

class WaterMark{    //水印文字    waterTexts = []    //需要添加水印的dom集合    needAddWaterTextElementIds = null    //保存添加水印的dom    saveNeedAddWaterMarkElement = []    //初始化    constructor(waterTexts,needAddWaterTextElementIds){        if(waterTexts && waterTexts.length != 0){            this.waterTexts = waterTexts        } else {            this.waterTexts = ['水印文字哈哈哈哈','2022-12-08']        }        this.needAddWaterTextElementIds = needAddWaterTextElementIds    }        //开始添加水印    startWaterMark(){        const self = this        if(this.needAddWaterTextElementIds){            this.needAddWaterTextElementIds.forEach((id)=>{                let el = document.getElementById(id)                self.saveNeedAddWaterMarkElement.push(el)            })        } else {            this.saveNeedAddWaterMarkElement = Array.from(document.getElementsByTagName('img'))        }        this.saveNeedAddWaterMarkElement.forEach((el)=>{            self.startWaterMarkToElement(el)        })    }    //添加水印到到dom对象    startWaterMarkToElement(el){        let nodeName = el.nodeName        if(['IMG','img'].indexOf(nodeName) != -1){            //图片,需要加载完成进行操作            this.addWaterMarkToImg(el)        } else {            //普通,直接添加            this.addWaterMarkToNormalEle(el)        }    }            //给图片添加水印    async addWaterMarkToImg(img){        if(!img.complete){            await new Promise((resolve)=>{                img.onload = resolve            })        }        this.addWaterMarkToNormalEle(img)    }        //给普通dom对象添加水印    addWaterMarkToNormalEle(el){        const self = this        let canvas = document.createElement('canvas')        canvas.width = el.width ? el.width : el.clientWidth        canvas.height = el.height ? el.height : el.clientHeight        let ctx = canvas.getContext('2d')        let maxSize = Math.max(canvas.height, canvas.width)        let font = (maxSize / 25)        ctx.font = font + 'px "微软雅黑"'        ctx.fillStyle = "rgba(195,195,195,1)"        ctx.textAlign = "left"        ctx.textBaseline = "top"        ctx.save()        let angle = -Math.PI / 10.0        //进行平移,计算平移的参数        let translateX = (canvas.height) * Math.tan(Math.abs(angle))        let translateY = (canvas.width - translateX) * Math.tan(Math.abs(angle))        ctx.translate(-translateX / 2.0, translateY / 2.0)        ctx.rotate(angle)        //起始坐标        let x = 0        let y = 0        //一组文字之间间隔        let sepY = (font / 2.0)        while(y < canvas.height){            //当前行的y值            let rowCurrentMaxY = 0            while(x {                    currentY += (index * (sepY + font))                    let rect = self.drawWater(ctx,text,x,y + currentY)                    let currentMaxX = (rect.x + rect.width)                    totleMaxX = (currentMaxX > totleMaxX) ? currentMaxX: totleMaxX                    rowCurrentMaxY = currentY                })                x = totleMaxX + 20            }            //重置x,y值            x = 0            y += (rowCurrentMaxY + (sepY + font + (canvas.height / 5)))        }        ctx.restore()        //添加canvas        this.addCanvas(canvas,el)    }    //绘制水印    drawWater(ctx,text,x,y){        //绘制文字        ctx.fillText(text,x,y)        //计算尺度        let textRect = ctx.measureText(text)        let width = textRect.width        let height = textRect.height        return {x,y,width,height}    }    //添加canvas到当前标签的父标签上    addCanvas(canvas,el){        //创建div(canvas需要依赖一个div进行位置设置)        let warterMarDiv = document.createElement('div')        //关联水印dom对象        el.warterMark = warterMarDiv        //添加样式        this.resetCanvasPosition(el)        //添加水印        warterMarDiv.appendChild(canvas)        //添加到父标签        el.parentElement.insertBefore(warterMarDiv,el)    }    //重新计算位置    resetCanvasPosition(el){        if(el.warterMark){            //设置父标签的定位            el.parentElement.style.cssText = `position: relative;`            //设施水印载体的定位            el.warterMark.style.cssText = 'position: absolute;top: 0px;left: 0px;pointer-events:none'        }    }}

用法

详解如何用JS实现覆盖水印效果
let waterMark = new WaterMark()waterMark.startWaterMark();

ctx.save()ctx.restore() 其实在这里的作用不是很大,但还是添加上了,目的是保存添加水印前的上下文,跟结束绘制后恢复水印前的上下文,这样,这些斜体字只在这两行代码之间生效,下面如果再绘制其他,那么,将不受影响。

防止蒙版水印遮挡底层按钮或其他事件,需要添加 pointer-events:none 属性到蒙版标签上。

添加水印的标签外需要添加一个 父标签 ,这个 父标签 的作用就是添加约束 蒙版canvas 的位置,这里想通过 MutationObserver 观察 body 的变化来进行更新 蒙版canvas 的位置,这个尝试失败了,因为复杂的布局只要变动会都在这个回调里触发。因此,直接在添加水印的标签外需要添加一个 父标签 ,用这个 父标签 来自动约束 蒙版canvas 的位置。

MutationObserver 逻辑如下,在监听回调里可以及时修改布局或者其他操作(暂时放弃)。

var MutationObserver = window.MutationObserver || window.webkitMutationObserver || window.MozMutationObserver;var mutationObserver = new MutationObserver(function (mutations) {    //修改水印位置})mutationObserver.observe(document.getElementsByTagName('body')[0], {    childList: true, // 子节点的变动(新增、删除或者更改)    attributes: true, // 属性的变动    characterData: true, // 节点内容或节点文本的变动    subtree: true // 是否将观察器应用于该节点的所有后代节点})

图片的大小只有在加载完成之后才能确定,所以,对于 IMG 的操作,需要观察它的 complete 事件。

三、总结与思考

canvas ctx.drawImage(img, 0, 0) 进行绘制,再将 canvas.toDataURL(‘image/png’) 生成的 url 加载到之前的图片上,也是一种方式,但是,有时候会因为图片的原因导致最后的合成图片的 base64 数据是空,所以,直接增加一个蒙版,本身只是为了显示,并不是要生成真正的合成图片。实现了简单的伪水印,没有特别复杂的代码,代码拙劣,大神勿笑[抱拳][抱拳][抱拳]

推荐学习:《JavaScript视频教程》

以上就是详解如何用JS实现覆盖水印效果的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/543153.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月9日 16:12:04
下一篇 2025年11月9日 16:13:15

相关推荐

  • Go Gorilla 框架会话管理:深度解析与实践指南

    本文旨在提供一份关于如何在 go 语言中使用 gorilla sessions 框架进行会话管理的全面教程。我们将详细探讨会话存储的初始化、会话的获取与设置、关键的 cookie 选项配置,以及确保会话数据正确保存到客户端浏览器的核心步骤,帮助开发者高效、安全地实现用户会话功能。 正文 1. 引言 …

    2025年12月16日
    000
  • Go Gorilla Sessions:深入理解与实践会话管理

    本教程详细讲解了如何在go语言中使用gorilla sessions框架进行会话管理。内容涵盖cookiestore的初始化、会话的获取与创建、会话值的设置与持久化,以及会话选项的配置,旨在帮助开发者构建安全、可靠的web应用会话机制。 1. 引言:Go Gorilla Sessions 简介 在W…

    2025年12月16日
    000
  • 自定义日志处理与用户行为分析:从文件系统到专业工具的最佳实践

    本教程探讨了自定义日志格式的解析、存储与分析策略。针对用户行为日志,文章指出传统文件系统存储的局限性,并推荐转向事件驱动的专业分析平台,如Mixpanel或Keen.io,以实现高效数据洞察与可视化。同时,也讨论了Unix工具、编程语言在日志解析中的应用场景,强调了可视化在理解数据中的核心作用。 在…

    2025年12月16日
    000
  • 如何在Go语言中使用Gorilla Sessions框架管理HTTP会话

    本教程全面指导如何在Go应用程序中利用Gorilla Sessions框架实现和管理HTTP会话。内容涵盖CookieStore的设置、会话的初始化与检索、会话值的设置与持久化,以及安全且健壮的会话选项配置,确保HTTP Cookie的正确处理。 1. Gorilla Sessions简介 HTTP…

    2025年12月16日
    000
  • 如何在Golang中处理跨域请求

    答案是通过设置CORS响应头或使用中间件处理跨域请求。Golang中可通过手动编写中间件或使用rs/cors库配置Access-Control-Allow-Origin、Methods、Headers等头部,正确响应预检请求,实现安全的跨域资源共享,生产环境应避免通配符并谨慎启用凭据支持。 在Gol…

    2025年12月16日
    000
  • 优化日志处理:从文件系统到事件分析的转变

    本文探讨了高效处理日志以理解用户行为的方法。针对传统文件系统日志存储在行为分析方面的局限性,我们推荐采用事件驱动的分析平台,如mixpanel或keen.io。这些平台通过结构化事件收集和强大的可视化功能,能够更直接、更深入地洞察用户行为,避免了手动解析和关联大量原始日志的复杂性。 在现代应用开发中…

    2025年12月16日
    000
  • 日志处理与用户行为分析:从传统解析到现代事件驱动方法

    本文深入探讨了自定义日志格式的解析与用户行为分析策略。针对传统文件系统组织日志的局限性,我们提出了一种现代的事件驱动方法。通过利用专业的事件分析平台和可视化%ignore_a_1%,可以更高效地收集、分析用户行为数据,并从中提取有价值的洞察,从而超越单纯的日志存储,实现数据驱动的决策。 在复杂的应用…

    2025年12月16日
    000
  • 如何使用Golang进行RPC错误处理

    Go RPC错误处理需区分调用失败与业务失败,前者通过返回error实现,后者应在Reply结构中嵌入错误字段如Error string或自定义AppError类型传递详细信息,同时避免panic并用defer+recover统一捕获异常,确保服务健壮性。 在使用 Golang 进行 RPC(远程过…

    2025年12月16日
    000
  • Golang html/template安全生成HTML示例

    使用 html/template 可自动转义变量防止 XSS,如 {{.Username}} 会转义恶意脚本;需插入可信 HTML 时可使用 template.HTML 类型,但必须确保内容安全;应避免手动拼接 HTML,而将原始数据交由模板处理,以保证各上下文正确转义。 Go 的 html/tem…

    2025年12月16日
    000
  • 如何在Golang中构建留言回复系统

    答案:使用Golang构建留言回复系统需定义树形结构的Comment模型,通过map存储并实现创建评论与构建评论树功能,结合net/http提供REST接口。 构建一个留言回复系统在Golang中并不复杂,关键是设计好数据结构和接口逻辑。系统需要支持用户发布留言、回复留言,并能按层级展示评论树。以下…

    2025年12月16日
    000
  • Golang如何构建小型在线问卷系统

    答案:用Golang构建小型在线问卷系统需定义问卷、问题和回答的结构体,使用内存存储并加锁保证并发安全,通过HTTP路由实现获取问卷、提交回答等接口,配合JSON数据交互。示例采用net/http和gorilla/mux处理请求,初期以map和切片存储数据,后续可扩展为SQLite持久化、用户认证和…

    2025年12月16日
    000
  • 如何在Golang中构建小型聊天室

    使用Go的goroutine和channel构建TCP聊天室,通过net.Listen监听端口,每个连接启动goroutine处理。2. 客户端结构体包含连接和用户名,用map维护在线用户,全局channel用于广播消息。3. handleConn读取客户端输入,首行为昵称并通知广播,后续消息带前缀…

    2025年12月16日
    000
  • 如何在Golang中实现数据分页显示

    分页通过page和page_size计算offset实现,数据库分页用LIMIT和OFFSET查询,内存分页用切片截取,需返回总条数、总页数等元信息。 在Golang中实现数据分页显示,核心是通过限制查询数量和偏移量来控制返回的数据范围。通常结合数据库查询(如MySQL、PostgreSQL)或内存…

    2025年12月16日
    000
  • 如何在Golang中实现验证码功能

    答案:使用github.com/mojocn/base64Captcha库可快速实现Golang图形验证码功能,1. 安装库后通过NewDriverDigit生成数字验证码配置;2. 调用Generate方法获取Base64编码的图片和唯一ID;3. 前端请求/api/captcha接口获取验证码图…

    2025年12月16日
    000
  • Golang并发WebSocket消息处理项目

    答案是利用goroutine和channel实现非阻塞通信,通过map和互斥锁管理连接,使用广播channel转发消息,读写分离确保并发安全,配合defer及时释放资源,避免内存泄漏。 用Go做并发WebSocket消息处理,核心是利用goroutine和channel实现非阻塞通信。关键不在技术选…

    2025年12月16日
    000
  • 在Go语言Web服务前置Nginx的优势

    在go语言web服务前置nginx作为反向代理,能带来显著的性能、安全和管理效益。nginx擅长处理日志记录、ssl/tls终止、http/2支持、内容压缩、http头部管理以及高效静态资源服务等“web服务器”任务,从而让go应用专注于业务逻辑,避免重复造轮子,构建更健壮、可扩展的系统。 将Ngi…

    2025年12月16日
    000
  • Go语言中获取变量类型字符串的实用方法

    在go语言中,获取变量的类型并以字符串形式打印是一个常见需求。本文将介绍如何使用`fmt.printf`函数的`%t`格式化动词来高效、简洁地实现这一目标,避免了类似javascript `typeof`或python `type`操作符的误区。通过一个简单的示例,读者将掌握在go中获取变量类型字符…

    2025年12月16日
    000
  • Golang如何开发在线计算器项目

    用Golang开发在线计算器需前后端协作:前端HTML页面通过fetch发送表达式,后端Go程序用net/http处理POST请求,借助govaluate解析计算并返回JSON结果,主函数注册/calculate路由和静态文件服务,项目结构清晰,可快速搭建运行。 用Golang开发一个在线计算器项目…

    2025年12月16日
    000
  • Golang如何构建简单的博客评论系统

    先定义评论结构体,包含ID、作者、内容和创建时间。使用切片和互斥锁在内存中存储评论,保证并发安全。通过net/http实现GET /comments获取所有评论,POST /comment提交新评论,处理JSON数据并校验字段。前端可嵌入HTML表单,用JavaScript调用API实现交互。核心是…

    2025年12月16日
    000
  • Go语言:获取变量类型字符串的两种实用方法

    本文深入探讨了go语言中获取变量类型并以字符串形式输出的两种主要方法。首先介绍如何利用`fmt.printf`函数的`%t`格式化动词进行快速打印,适用于调试和日志记录。随后,详细阐述了如何借助`reflect`包的`typeof`函数来编程获取变量的类型字符串,包括`string()`和`name…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信