HarmonyOS 实战:给笔记应用加防截图水印
最近在做笔记类应用时,遇到一个头疼的需求:防止用户内容被非法截图传播。思来想去,加水印是个直接有效的方案。研究了 HarmonyOS 的开发文档后,发现用 Canvas 配合布局组件能轻松实现动态水印效果。今天就来聊聊如何给笔记页面加上「会呼吸」的用户专属水印,顺便分享几个开发时踩过的坑。
一、需求拆解:什么样的水印防截图最有效?
我们的目标很明确:在笔记浏览页面覆盖一层半透明水印,内容包含用户 ID+实时时间戳,且满足以下条件:
- 斜向排列:防止截图后通过简单裁剪去除
- 动态更新:每分钟刷新时间戳,增加追踪难度
- 性能无感:不影响页面滑动和交互
- 全局覆盖:适配不同屏幕尺寸和旋转方向
二、核心实现:用canvas实现动态水印
1. 搭建水印组件骨架
首先封装一个 UserWatermark
组件,基于 HarmonyOS 的 Canvas 实现自绘。这里有个关键细节:通过 hitTestBehavior
设置水印层透明,避免阻挡用户点击笔记内容。
// components/Watermark.ets
@Component
export struct UserWatermark {@State userId = 'user001' // 从账号服务获取的动态用户IDprivate context = new CanvasRenderingContext2D(new RenderingContextSettings(true))private timestamp = new Date().toLocaleString() // 实时时间戳build() {Canvas(this.context).width('100%').height('100%').hitTestBehavior(HitTestMode.Transparent) // 重点!不影响触摸事件.onReady(() => this.drawWatermark())}// 初始化绘制private drawWatermark() {this.updateTimestamp() // 先更新时间this.context.clearRect(0, 0, this.context.width, this.context.height)this.setWatermarkStyle()this.drawWatermarkGrid()}
}
2. 动态时间戳实现:每分钟刷新一次
为了避免高频重绘影响性能,选择每分钟更新一次时间戳。这里用 setInterval
配合状态变量触发重绘:
// 组件生命周期钩子
aboutToAppear() {this.updateTimestamp() // 初始化时间setInterval(() => {this.timestamp = new Date().toLocaleString() // 更新时间this.drawWatermark() // 触发画布重绘}, 60 * 1000) // 每分钟执行一次
}// 时间格式化方法(可根据需求调整)
private updateTimestamp() {const now = new Date()this.timestamp = `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, '0')}-${String(now.getDate()).padStart(2, '0')} ${String(now.getHours()).padStart(2, '0')}:${String(now.getMinutes()).padStart(2, '0')}`
}
3. 斜向水印的坐标计算「玄学」
实现斜向排列的关键是坐标系旋转和平移。这里踩过最大的坑是旋转后原点位置的变化,调试了半小时才发现需要将原点移到左下角:
private drawWatermarkGrid() {const text = `用户ID:${this.userId} ${this.timestamp}`const font = '14vp sans-serif'const angle = -25 * Math.PI / 180 // 倾斜25度// 计算文本尺寸this.context.font = fontconst { width: textWidth, height: textHeight } = this.context.measureText(text)// 坐标系变换:先平移到左下角,再旋转this.context.save()this.context.translate(0, this.context.height) // 原点移至左下角this.context.rotate(angle) // 逆时针旋转25度// 计算行列间隔(1.5倍文本尺寸避免重叠)const colGap = textWidth * 1.5const rowGap = textHeight * 1.5const cols = Math.ceil(this.context.width / colGap)const rows = Math.ceil(this.context.height / rowGap)// 绘制网格for (let i = 0; i < cols; i++) {for (let j = 0; j < rows; j++) {const x = i * colGapconst y = j * rowGapthis.context.fillText(text, x, -y) // y轴反转(因为原点在左下角)}}this.context.restore() // 恢复原始坐标系
}
4. 页面集成:用 Stack 实现水印层覆盖
将水印组件与笔记内容层叠加,推荐使用 Stack
布局,清晰且性能稳定:
// pages/NoteDetail.ets
@Entry
@Component
struct NoteDetail {private noteContent = '这里是用户的笔记正文...'build() {Stack() {// 笔记内容层Column().padding(24).spacing(16).text(this.noteContent).fontSize(16).lineHeight(24)// 水印层(覆盖在内容上方)UserWatermark()}.backgroundColor(Color.White).width('100%').height('100%')}
}
三、性能优化
-
离屏渲染优化
如果遇到页面卡顿,可以尝试将Canvas
替换为OffscreenCanvas
进行离屏绘制,减少主线程压力:// 离屏画布版本(适用于复杂场景) private offscreenCanvas = new OffscreenCanvas() private offscreenContext = this.offscreenCanvas.getContext('2d')!// 在draw方法中使用offscreenContext绘制,最后同步到主画布 this.context.drawImage(this.offscreenCanvas, 0, 0)
-
文本测量缓存
重复计算文本宽度会影响性能,因此将measureText
的结果缓存:private textMetrics: TextMetrics | null = nullprivate getTextSize(text: string) {if (!this.textMetrics || this.textMetrics.text !== text) {this.textMetrics = this.context.measureText(text)}return this.textMetrics }
-
触摸事件优化
通过hitTestBehavior: HitTestMode.Transparent
让水印层完全透明,触摸事件直接穿透到下层内容,不影响用户操作。
四、从页面到全场景的水印方案
如果你的应用需要支持更多场景,还可以参考官方文档中的其他能力:
- 图片水印:用
OffscreenCanvas
实现本地图片加水印(适合用户保存笔记截图时自动加水印) - PDF 水印:通过
pdfService
模块给导出的 PDF 文档添加水印(企业需求必备) - 动态变色:根据页面主题切换水印颜色(浅色/深色模式适配)
五、那些让我半夜睡不着的细节
-
旋转方向的坑
坐标系默认顺时针旋转,想实现「向左倾斜」需要用负数角度(如-25度
),刚开始用正数导致水印方向搞反。 -
设备适配问题
不同设备的vp
单位换算有差异,建议统一处理尺寸问题。 -
性能监控
用 DevEco Studio 的「性能调优」工具监控onReady
和draw
方法的执行时间,确保单次绘制不超过 16ms(60fps标准)。
六、总结:水印背后的安全哲学
加水印本质是一种「威慑性防护」,它不能完全阻止截图,但能大大增加内容泄露后的追溯成本。在实际开发中,建议结合以下策略:
- 前端水印:防止非授权截图传播
- 后端日志:记录用户操作时间线
- 数据加密:敏感内容本地加密存储
技术之外,更重要的是平衡用户体验与安全需求——毕竟,没有人喜欢被密密麻麻的水印「包围」。通过透明度调整(建议 opacity: 0.15-0.2
)和合理的排列间隔,完全可以做到「水印可见但不干扰阅读」。
如果你在开发中遇到其他有趣的场景,欢迎在评论区交流~ 一起用技术让内容安全更优雅一点~ 💻✨
相关文章:
HarmonyOS 实战:给笔记应用加防截图水印
最近在做笔记类应用时,遇到一个头疼的需求:防止用户内容被非法截图传播。思来想去,加水印是个直接有效的方案。研究了 HarmonyOS 的开发文档后,发现用 Canvas 配合布局组件能轻松实现动态水印效果。今天就来聊聊如何给笔记页面加上…...

如何轻松地将文件从 PC 传输到 iPhone?
传统上,您可以使用 iTunes 将文件从 PC 传输到 iPhone,但现在,使用 iTunes 已不再是唯一的选择。现在有多种不同且有效的方法可以帮助您传输文件。在今天的指南中,您可以找到 8 种使用或不使用 iTunes 传输文件的方法,…...
前端面试二之运算符与表达式
目录 1.JavaScript 中的 和 运算符 2.|| (逻辑或) 运算符 与 ES6 默认参数的区别 与 ?? (空值合并运算符) 的区别 3.?.(可选链)运算符 (1). 安全访问深层嵌套属性 (2). 安全调用可能不存在的函数 (3). 安全访问数组元素 4.展开运算符 (..…...
【运维实战】使用Nvm配置多Node.js环境!
背景 新项目 使用Node.js-v16.17.1旧项目 使用Node.js- v14.18.0 【且依赖于node-saas模块,根据 node-sass 的官方文档,目前最新版本的 node-sass(即 v5.0.0)支持的 Node.js 版本范围是 Node.js 10.x、Node.js 12.x、Node.js 14.…...

Bresenham算法
一 Bresenham 绘直线 使用 Bresenham 算法,可以在显示器上绘制一直线段。该算法主要思想如下: 1 给出直线段上两个端点 ,根据端点求出直线在X,Y方向上变化速率 ; 2 当 时,X 方向上变化速率快于 Y 方向上变化速率&am…...

【从GEO数据库批量下载数据】
从GEO数据库批量下载数据 1:进入GEO DataSets拿到所需要下载的数据的srr.list,上传到linux, 就可以使用prefetch这个函数来下载 2:操作步骤如下: conda 安装sra-tools conda create -n sra-env -c bioconda -c co…...

day 44
使用DenseNet预训练模型对cifar10数据集进行训练 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms, models from torch.utils.data import DataLoader import matplotlib.pyplot as plt import os# 设置中文字体…...
鸿蒙OSUniApp开发跨平台AR扫描识别应用:HarmonyOS实践指南#三方框架 #Uniapp
UniApp开发跨平台AR扫描识别应用:HarmonyOS实践指南 前言 随着增强现实(AR)技术在移动应用中的广泛应用,越来越多的开发者需要在跨平台应用中实现AR功能。本文将深入探讨如何使用UniApp框架开发一个高性能的AR扫描识别应用&…...

NER实践总结,记录一下自己实践遇到的各种问题。
更。 没卡,跑个模型休息好几天,又闲又急。 一开始直接套用了别人的代码进行实体识别,结果很差,原因是他的词表没有我需要的东西,我是用的医学文本。代码直接在github找了改的,用的是BERT的Chinese版本。 然…...

微信小程序实现运动能耗计算
微信小程序实现运动能耗计算 近我做了一个挺有意思的微信小程序,能够实现运动能耗的计算。只需要输入性别、年龄、体重、运动时长和运动类型这些信息,就能算出对应的消耗热量。 具体来说,在小程序里,性别不同,身体基…...

iTunes 无法备份 iPhone:10 种解决方法
Apple 设备是移动设备市场上最先进的产品之一,但有些人遇到过 iTunes 因出现错误而无法备份 iPhone 的情况。iTunes 拒绝备份 iPhone 时,可能会令人非常沮丧。不过,幸运的是,我们有 10 种有效的方法可以解决这个问题。您可以按照以…...
施耐德特价型号伺服电机VIA0703D31A1022、常见故障
⚙️ 一、启动类故障 电机无法启动 可能原因:电源未接通、制动器未释放、接线错误或控制器故障。解决措施: 检查电源线路及断路器状态;验证制动器是否打开(带制动器型号);核对电机与控制器…...

LangChain4J 使用实践
这里写目录标题 大模型应用场景:创建一个测试示例AIService聊天记忆实现简单实现聊天记录记忆MessageWindowChatMemory实现聊天记忆 隔离聊天记忆聊天记忆持久化 添加AI提示词 大模型应用场景: 创建一个测试示例 导入依赖 <dependency><groupI…...
慢SQL调优(二):大表查询
最近在工作中写SQL出现几次慢SQL的BUG,总结下来归根到底就是因为大表的原因~这表有多大呢,执行 select COUNT(1) FROM position 是出不来结果滴,每天保底新增1000条数据,可想而知有多大了,所以多次踩坑了这张表。所以…...

【C++】—— 从零开始封装 Map 与 Set:实现与优化
人生的态度是,抱最大的希望,尽最大的努力,做最坏的打算。 —— 柏拉图 《理想国》 目录 1、理论基石——深度剖析 BSTree、AVLTree 与 RBTree 的概念区别 2、迭代器机制——RBTree 迭代器的架构与工程实现 3、高级容器设计——Map 与 Set…...

内网穿透之Linux版客户端安装(神卓互联)
选择Linux系统版本 获取安装包 :https://www.shenzhuohl.com/download.html 这里以Ubuntu 18.04为例,其它版本方法类似 登录Ubuntu操作系统: 打开Ubuntu系统终端,更新版本 apt-get update 安装运行环境: 安装C 运…...

开疆智能Profinet转Profibus网关连接CMDF5-8ADe分布式IO配置案例
本案例是客户通过开疆智能研发的Profinet转Profibus网关将PLC的Profinet协议数据转换成IO使用的Profibus协议,操作步骤如下。 配置过程: Profinet一侧设置 1. 打开西门子组态软件进行组态,导入网关在Profinet一侧的GSD文件。 2. 新建项目并…...

华为云Flexus+DeepSeek征文|Flexus云服务器单机部署+CCE容器高可用部署快速搭建生产级的生成式AI应用
前引: 在AI技术高速演进的浪潮中,如何快速、高效、安全地搭建一个大模型应用平台,成为开发者和企业关注的焦点。近日,华为云推出的Flexus云服务器配合CCE容器引擎和Dify LLM应用开发平台,带来了极具吸引力的解决方案。…...
扫地机产品--材质传感器算法开发与虚拟示波器
扫地机产品–材质传感器算法开发与虚拟示波器 文章目录 扫地机产品--材质传感器算法开发与虚拟示波器**一、材质传感器的工作原理**二、核心功能与应用场景三、技术参数与产品示例四.MCU 与压电陶瓷超声波的材质检测技术方案实现原理分析4.1 超声波原理4.2表面类型检测4.3 超声…...
[蓝桥杯]上三角方阵
上三角方阵 题目描述 方阵的主对角线之上称为"上三角"。 请你设计一个用于填充 nn 阶方阵的上三角区域的程序。填充的规则是:使用 1,2,3.... 的自然数列,从左上角开始,按照顺时针方向螺旋填充。 例如&am…...

60天python训练计划----day44
DAY 44 预训练模型 知识点回顾: 预训练的概念常见的分类预训练模型图像预训练模型的发展史预训练的策略预训练代码实战:resnet18 一、预训练的概念 我们之前在训练中发现,准确率最开始随着epoch的增加而增加。随着循环的更新,参数…...

【JAVA版】意象CRM客户关系管理系统+uniapp全开源
一.介绍 CRM意象客户关系管理系统,是一个综合性的客户管理平台,旨在帮助企业高效地管理客户信息、商机、合同以及员工业绩。系统通过首页、系统管理、工作流程、审批中心、线索管理、客户管理、商机管理、合同管理、CRM系统、数据统计和系统配置等模块&…...

API异常信息如何实时发送到钉钉
#背景 对于一些重要的API,开发人员会非常关注API有没有报错,为了方便开发人员第一时间获取错误信息,我们可以使用插件来将API报错实时发送到钉钉群。 接下来我们就来实操如何实现 #准备工作 #创建钉钉群 如果已有钉钉群,可以跳…...

Python爬虫(48)基于Scrapy-Redis与深度强化学习的智能分布式爬虫架构设计与实践
目录 一、背景与行业痛点二、核心技术架构设计2.1 分布式爬虫基础架构2.2 深度强化学习模块 三、生产环境实践案例3.1 电商价格监控系统3.2 学术文献采集系统 四、高级优化技术4.1 联邦学习增强4.2 神经架构搜索(NAS) 五、总结🌈Python爬虫相…...
AtCoder Beginner Contest 407 E - Most Valuable Parentheses
AtCoder Beginner Contest 407 E - Most Valuable Parentheses E - Most Valuable Parentheses 反悔贪心算法 性质: 假设长度为 n n n, n ≡ 0 ( m o d 2 ) n \equiv 0 \pmod{2} n≡0(mod2) 的括号序列是合法的,那么有 n 2 \frac{n}{2}…...

(1-6-3)Java 多线程
目录 0.知识拓扑 1. 多线程相关概念 1.1 进程 1.2 线程 1.3 java 中的进程 与 线程概述 1.4 CPU、进程 与 线程的关系 2.多线程的创建方式 2.1 继承Thread类 2.2 实现Runnable接口 2.3 实现Callable接口 2.4 三种创建方式对比 3.线程同步 3.1 线程同步机制概述 …...

java31
1.网络编程 三要素: 网址实质上就是ip InetAddress: UDP通信程序: 多个接收端的地址都要加入同一个组播地址,这样发送端发信息,全部接收端都能接受到数据 广播的代码差不多,就是地址不一样而已 TCP通信程序…...
多模态之智能数字人
多模态下智能数字人的开发是一个复杂且系统性的工程,它融合了人工智能(AI)、计算机图形学、自然语言处理(NLP)、语音技术、计算机视觉(CV)等多个前沿领域。 多模态下智能数字人的开发流程规范 目标: 构建一个能够理解并生成多模态信息(文本、语音、视觉等),具备智…...

界面组件DevExpress WPF中文教程:Grid - 如何识别行和卡片?
DevExpress WPF拥有120个控件和库,将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpress WPF能创建有着强大互动功能的XAML基础应用程序,这些应用程序专注于当代客户的需求和构建未来新一代支持触摸的解决方案。 无论是Office办公软件…...

【HarmonyOS Next之旅】DevEco Studio使用指南(三十)
目录 1 -> 部署云侧工程 2 -> 通过CloudDev面板获取云开发资源支持 3 -> 通用云开发模板 3.1 -> 适用范围 3.2 -> 效果图 4 -> 总结 1 -> 部署云侧工程 可以选择在云函数和云数据库全部开发完成后,将整个云工程资源统一部署到AGC云端。…...