PDF 转 HTML5 —— HTML5 填充图形不支持 Even-Odd 奇偶规则?(第二部分)

这是关于该主题的第二部分。如果你还没有阅读第一部分,请先阅读,以便理解“绕组规则”的问题。

快速回顾一下:HTML5 只支持 Non-Zero(非零)绕组规则,而 PDF 同时支持 Non-Zero 和 Even-Odd(奇偶)两种规则。
这意味着我们必须对使用了 Even-Odd 填充规则的图形做些处理,否则它们在 HTML5 中可能无法正确显示。

为了演示这个问题,我会用以下这个例子:

看上去只是个红色的圆?
我第一次看到的时候也是这么想的。直到我把填充改为描边模式,才意识到它实际上是什么。

为了帮助理解发生了什么,我们再来看一张图,加上箭头,显示路径的绘制方向。

我们首先想到的是,可能需要改变图形的绘制方式。最直接的想法就是改变路径的方向,也就是“转换”成 Non-Zero 绕组规则的等价图形。

交替使用顺时针和逆时针方向可行吗?
值得一试,看看会发生什么。反正只需要一分钟时间试一下。

结果证明,这其实挺头疼的。如果只是用 lineTo 画个方框还好说,但涉及贝塞尔曲线的话,情况就复杂得多。

如果你只是简单地反转绘制指令的顺序,就会得到像这样的问题:

控制点的顺序会影响曲线的绘制方式。那么,我们能否也反转控制点的顺序呢?

哦——这样也不太对。我们使用的是 bezierCurveTo,它的意思是:从当前位置(由上一个绘图命令决定)开始,通过两个控制点绘制一段曲线,终点是我们指定的位置。这就意味着我们还必须反向地重新组织控制点和终点的顺序,这样曲线才能按正确的方式被绘制出来。

做对了之后,我们就能反转路径方向,让图形按预期绘制出来:

那么问题解决了吗?没有。

对于我们的“禁止通行”标志来说,或许可以这样处理——我们可以手动控制路径的顺序和方向。但在现实世界中,这种做法不能适用于通用场景。我们不可能为每个具体的 PDF 文件都手动处理,我们需要的是一套普适的代码逻辑,让它能应对任何 PDF 文件。

这也就意味着交替改变路径方向并不能真正解决问题。比如看一下我们禁止通行标志在这种方式下的效果:

我的下一个想法是:用背景色填充被遮挡的区域。但这也行不通,因为路径的绘制顺序还是关键。即使我们把两个 “D” 字母填充为白色,当外面的圆形被填充时,它还是会把我们之前的工作覆盖掉。

而且如果两个形状只是部分重叠,那么只有重叠部分才需要“未填充”。更关键的是,这些“未填充”的部分是真正的透明区域,如果你用背景色填充,就会失去透明性,导致我们无法在后面放其他图层,比如显示“你不可以做这件事”的提示。

那怎么办?
如果不能改变绘制顺序,也不能改变路径方向,也不能通过填色伪装,我们还能怎么改造图形,让它在使用 Non-Zero 绕组规则时正常显示?

或许我们可以非常巧妙地将图形切割成许多小块,然后分析每一块该如何绘制,再分别填充这些小块。

像这样:

_____
|   __|__
|__|__|  |
   |_____|

 _____
|   __|
|__|
    __
   |__|
       __
    __|  |
   |_____|
 

但这就太复杂了。对于我们的需求来说,有点大材小用了。而且,这种做法计算量大,会降低我们转换器的性能。更现实的是,我也没那么聪明,写不出来这样的算法。

其实从我开始研究这些形状开始,脑海里就一直有一个小声音在说:
“要不,我们干脆把这些图形输出成图片得了?”

确实,这样做是可行的,但并不理想。实际上,使用 Non-Zero 和 Even-Odd 绕组规则绘制图形,最终渲染结果真正不同的情况并不多。

所以,如果我们每次遇到 Even-Odd 绘制规则就输出成图片,那会产生大量没必要的图像文件。

这也不理想,因为矢量图形本身非常好用:缩放不会失真,从 PDF 转到 HTML5 非常快速,占用空间也远远小于图片。

那我们还能怎么办?

我们只能妥协。
最合理的做法就是:设定一套规则,只有当图形符合某些特定条件时,才把它输出为图片。
其余情况我们仍然尽可能用原始的路径绘制方式进行呈现。这样,在保持性能和显示效果之间取得一个平衡。

我们的主页:PDF 转 HTML5、Java 图像库、Java PDF SDK - IDRsolutions

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/90608.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/90608.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器学习 KNN 算法,鸢尾花案例

目录 一.机器学习概述 二.人工智能的两大方向 三.KNN算法介绍 1.核心思想:“物以类聚,人以群分” 2.算法步骤 四.KNN算法实现 1.安装scikit-learn库 2.导入knn用于分类的类KNeighborsClassifier 3.设置KNeighborsClassifier的相关参数 4.训练模…

强化学习(第三课第三周)

文章目录强化学习(第三课第三周)一、以火星探测器为例说明强化学习的形式化表示二、强化学习中的回报三、强化学习算法的目标(一)马尔可夫决策过程(二)状态动作价值函数(四)使用Bell…

星痕共鸣数据分析2

今天实验内容是攻击力部分 1.思路 由于昨天数据分析出了一个函数 这个函数可以把奇怪的字节变成正常的数字 int parse_varint(unsigned const char* data, int count) {int value 0;int shift 0;for (int i 0; i < count; i) {unsigned char byte data[i];value | ((byt…

强化学习新发现:仅需更新5%参数的稀疏子网络可达到全模型更新效果

摘要&#xff1a;强化学习&#xff08;RL&#xff09;已成为大语言模型&#xff08;LLM&#xff09;在完成预训练后与复杂任务及人类偏好对齐的关键步骤。人们通常认为&#xff0c;要通过 RL 微调获得新的行为&#xff0c;就必须更新模型的大部分参数。本研究对这一假设提出了挑…

electron 使用记录

目录 代理设置以打包成功 参考文档 代理设置以打包成功 参考文档 使用 JavaScript、HTML 和 CSS 构建跨平台桌面应用 |电子 --- Build cross-platform desktop apps with JavaScript, HTML, and CSS | Electron

Spring boot Grafana优秀的监控模板

JVM (Micrometer) | Grafana Labs 1 SLS JVM监控大盘 | Grafana Labs Spring Boot 2.1 Statistics | Grafana Labs springboot granfana 监控接口指定接口响应的 在Spring Boot应用中&#xff0c;使用Grafana进行监控通常涉及以下几个步骤&#xff1a; 设置Prometheus作…

LeetCode11~30题解

LeetCode11.盛水最多的容器&#xff1a; 题目描述&#xff1a; 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器…

计算机结构-逻辑门、存储器、内存、加法器、锁存器、程序计数器

逻辑门 逻辑门简单地理解即通过特定的条件实现与、或、非、异或等相关逻辑二极管 这些最基础的逻辑门都是通过电路元器件进行搭建的&#xff0c;即半导体材料搭建的二极管二极管有个特点&#xff0c;一定条件下才可以导通&#xff0c;即得接对正负极&#xff0c;具体的原理可以…

连锁店铺巡查二维码的应用

在连锁店铺的运营管理中&#xff0c;巡查工作是保障各门店规范运作、提升服务质量的关键环节。巡查二维码的出现&#xff0c;为这一环节带来了高效、便捷且规范的解决方案&#xff0c;其应用场景广泛&#xff0c;优势显著。在如今的繁杂且效果参差不齐电子二维码市场中&#xf…

各种前端框架界面

前端技术更新迭代很快&#xff0c;已经有不少新的前端框架问世&#xff0c;而且像geeker-admin风格的界面设计也挺不错的。 今天去面试了前端开发岗位&#xff0c;感觉希望不大。毕竟中间空了一段时间没接触&#xff0c;得赶紧把新的知识点补上&#xff0c;这样哪怕是居家办公也…

DApp 开发者 学习路线和规划

目录 🚀 一、学习路线图 阶段 1:基础知识(1~2 周) 阶段 2:智能合约开发(3~4 周) 阶段 3:前端与区块链交互(2~3 周) 阶段 4:进阶与生态系统(持续学习) 📅 二、学习规划建议(3~4 个月) 🧰 三、工具推荐 💡 四、附加建议 🚀 一、学习路线图 阶段 …

数据结构 二叉树(3)---层序遍历二叉树

在上篇文章中我们主要讲了关于实现二叉树的内容&#xff0c;包括遍历二叉树&#xff0c;以及统计二叉树等内容。而在这篇文章中我们将详细讲解一下利用队列的知识实现层序遍历二叉树。那么层序遍历是什么&#xff1f;以及利用队列遍历二叉树又是怎么遍历的&#xff1f;下面让我…

【橘子分布式】gRPC(番外篇-拦截器)

一、简介 我们之前其实已经完成了关于grpc的一些基础用法&#xff0c;实际上还有一些比较相对进阶的使用方式。比如&#xff1a; 拦截器&#xff1a;包括客户端和服务端的拦截器&#xff0c;进而在每一端都可以划分为流式的拦截器和非流式的拦截器。和以前我们在spring web中的…

深入探索嵌入式仿真教学:以酒精测试仪实验为例的高效学习实践

引言&#xff1a;嵌入式技术普及下的教学革新 嵌入式系统作为现代科技的核心驱动力&#xff0c;其教学重要性日益凸显。然而&#xff0c;传统硬件实验面临设备成本高、维护难、时空受限等挑战。如何突破这些瓶颈&#xff0c;实现高效、灵活、专业的嵌入式教学&#xff1f;本文将…

三种深度学习模型(GRU、CNN-GRU、贝叶斯优化的CNN-GRU/BO-CNN-GRU)对北半球光伏数据进行时间序列预测

代码功能 该代码实现了一个光伏发电量预测系统&#xff0c;采用三种深度学习模型&#xff08;GRU、CNN-GRU、贝叶斯优化的CNN-GRU/BO-CNN-GRU&#xff09;对北半球光伏数据进行时间序列预测对北半球光伏数据进行时间序列预测&#xff0c;并通过多维度评估指标和可视化对比模型性…

PostgreSQL对象权限管理

本文记述在postgreSQL中对用户/角色操作库、模式、表、序列、函数、存储过程的权限管理针对数据库的授权 授权&#xff1a;grant 权限 on database 数据库 to 用户/角色; 撤权&#xff1a;revoke 权限 on database 数据库 from 用户/角色; 针对模式的授权 授权&#xff1a;gran…

Wordpress主题配置

一、下载主题 主题下载地址&#xff1a;https://www.iztwp.com/tag/blog-theme 二、主题安装 三、上传主题安装即可 四、安装完成启动主题

lock 和 synchronized 区别

1. 引言 在多线程编程中&#xff0c;我们经常需要确保某些代码在同一时刻只由一个线程执行。这种机制通常叫做“互斥锁”或“同步”。Java 提供了两种主要的同步机制&#xff1a;synchronized 关键字和 Lock 接口。尽管它们的作用相似&#xff0c;都用于实现线程的同步&#xf…

Tkinter - Python图形界面开发指南

作者&#xff1a;唐叔在学习 专栏&#xff1a;唐叔学python 标签&#xff1a;Python GUI编程 Tkinter教程 图形界面开发 Python实战 界面设计 事件监听 Python入门 唐叔Python 编程学习 软件开发 文章目录一、Tkinter是什么&#xff1f;为什么选择它&#xff1f;二、Tkinter基础…

Java基础day15

目录 一、Java集合简介 1.什么是集合&#xff1f; 2.集合接口 3.小结 二、List集合 1.List集合简介 三、ArrayList容器类 1.初始化 1.1无参初始化 1.2有参初始化 2.数据结构 3.常用方法 3.1增加元素 3.2查找元素 3.3 修改元素 3.4 删除元素 3.5 其他方法 4.扩…