【绘制】HTML5 Canvas 中路径(path)、描边(stroke)与填充(fill)详细介绍(图文、代码示例)

本文主要是介绍【绘制】HTML5 Canvas 中路径(path)、描边(stroke)与填充(fill)详细介绍(图文、代码示例),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我的处女作《Canvas系列教程》在我的Github上正在连载更新,希望能得到您的关注和支持,让我有更多的动力进行创作。

教程介绍、教程目录等能在README里查阅。

传送门:https://github.com/827652549/CanvasStudy

目录

 

引入

案例

介绍

在线演示

效果图

代码

路径

子路径 

填充路径中的“非零环绕规则”

“非零环绕规则”的判断过程


引入

至今为止,我们所绘制的唯一图形,就是通过strokeRect()绘制矩形,也使用fillRect()进行填充。这两个方法都是立即生效的,的。

立即生效的Canvas API中的方法:

  • strokeRect()
  • fillRect()
  • strokeText()
  • fillText()

绘图环境中,可以绘制更为复杂的图形,如贝塞尔曲线等,这些方法都是基于路径(path)的。

大多数绘制系统,如SVG、Apple的Cocoa框架、Adobe illustrator等,都是基于路径的,使用这些绘制系统时,都是先定义一个路径,然后对其描边或填充。图示演示了三种绘制方式(描边、填充、描边和填充)。

案例

介绍

分别用矩形、开放路径、闭合路径演示三种不同的绘制方式。

在线演示

https://827652549.github.io/Canvas/Unit2/PathStrokeFill.html

效果图

闭合路径的代码实现比开放路径多写了一条closePath,用于将此时的路径从当前点回到起点。

代码

https://github.com/827652549/CanvasStudy/blob/master/Unit2/PathStrokeFill.html

路径

CanvasRenderingContext2D之中与路径有关的方法
方法描述
arc(x,y,r,sAngle,eAngle,true)

在当前路径中添加一段表示圆弧或圆形的子路径。五个参数分别是:

x坐标,y坐标,半径,起始角,结束角,(顺/逆时针)

beginPath()清除所有子路径,开始一段新路径
closePath()将当前路径位置连接到起点
fill()使用fillStyle对当前路径内部进行填充
rect(x,y,width,height)创建一个矩形
stroke()使用strokeStyle对当前路径进行描边

路径与隐形墨水

这是一个恰当对比喻,用来说明“创建路径随后对其进行描边或填充”,我们将这个操作比作“使用隐形墨水来绘图”。

你用隐形墨水所控制对内容不会立即显示出来,但你可以调节隐形墨水的颜色,笔尖宽度等等,最后用“特殊的方法(fill或stroke)”将图像显示出来。

子路径 

在某一时刻,canvas之中只能有一条路径存在,Canvas规范将其称为“当前路径(current path)”,这条路径可以包含很多子路径(subpath):有两个或更多的点组成。

为了理解什么是子路径,我们先来看一段代码:

context.beginPath();context.rect(10,10,100,100);
context.stroke();context.beginPath();context.rect(30,30,100,100);
context.stroke();

首先调用一次beginPath()开启一段新路径,该方法会将当前路径中所有子路径都清除掉,然后调用rect()方法,该方法向当前路径中添加了一个含有4个点的子路径,最后调用stroke()方法描边显形。

接下来再次调用了beginPath()方法,清除了上次调用rect()方法所创建的子路径(指那个矩形路径),然后再添加了另一个矩形子路径,最后进行描边显形。

如果我们第二次不调用beginPath()方法,那么此时路径便有两个矩形,那么第二次stroke()时便又会重新绘制一遍第一个矩形。当然,我们也可以同时删除第一个stroke()和第二个beginPath()来达到同样的效果,但为了让你理解,我有意这么做。

个人认为,子路径(subpath)不应该这么翻译,容易与继承关系中的“父子”中的“子”混淆,在Canvas这个应用场景中,被翻译成“补充路径”应该更容易理解。个人愚见,还请多指教。

填充路径中的“非零环绕规则”

如果当路径是循环的,或者包含多个相交的子路径,那fill()就会按照“非零环绕规则(nonzero winding rule)”来判断填充哪些区域。

“非零环绕规则”的判断过程

  1. 对于路径中的任意给定区域,在区域内画一条足够长的线段,使线段终点完全落于路径范围之外。(图中三个箭头表示此过程)
  2. 计数器初始化为0,对于每一条线段来说,每当线段与路径顺时针部分相交时+1,与路径逆时针部分相交时-1。
  3. 如果计数器最终值不是0,则证明在内部,如果是0,则证明在外部。仅填充在内部的区域。

以上图为例

上箭头与逆时针相交,计数器结果-1,填充;

右箭头与两个逆时针相交,结果-2,填充;

左箭头同时与一个顺时针和逆时针路径相交,结果为0,不填充。

这篇关于【绘制】HTML5 Canvas 中路径(path)、描边(stroke)与填充(fill)详细介绍(图文、代码示例)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/253486

相关文章

C++快速排序超详细讲解

《C++快速排序超详细讲解》快速排序是一种高效的排序算法,通过分治法将数组划分为两部分,递归排序,直到整个数组有序,通过代码解析和示例,详细解释了快速排序的工作原理和实现过程,需要的朋友可以参考下... 目录一、快速排序原理二、快速排序标准代码三、代码解析四、使用while循环的快速排序1.代码代码1.由快

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

Spring Boot拦截器Interceptor与过滤器Filter详细教程(示例详解)

《SpringBoot拦截器Interceptor与过滤器Filter详细教程(示例详解)》本文详细介绍了SpringBoot中的拦截器(Interceptor)和过滤器(Filter),包括它们的... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)详细教程1. 概述1

mybatis-plus 实现查询表名动态修改的示例代码

《mybatis-plus实现查询表名动态修改的示例代码》通过MyBatis-Plus实现表名的动态替换,根据配置或入参选择不同的表,本文主要介绍了mybatis-plus实现查询表名动态修改的示... 目录实现数据库初始化依赖包配置读取类设置 myBATis-plus 插件测试通过 mybatis-plu

使用Dify访问mysql数据库详细代码示例

《使用Dify访问mysql数据库详细代码示例》:本文主要介绍使用Dify访问mysql数据库的相关资料,并详细讲解了如何在本地搭建数据库访问服务,使用ngrok暴露到公网,并创建知识库、数据库访... 1、在本地搭建数据库访问的服务,并使用ngrok暴露到公网。#sql_tools.pyfrom

Qt把文件夹从A移动到B的实现示例

《Qt把文件夹从A移动到B的实现示例》本文主要介绍了Qt把文件夹从A移动到B的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录如何移动一个文件? 如何移动文件夹(包含里面的全部内容):如何删除文件夹:QT 文件复制,移动(

Flask 验证码自动生成的实现示例

《Flask验证码自动生成的实现示例》本文主要介绍了Flask验证码自动生成的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习... 目录生成图片以及结果处理验证码蓝图html页面展示想必验证码大家都有所了解,但是可以自己定义图片验证码

Windows系统下如何查找JDK的安装路径

《Windows系统下如何查找JDK的安装路径》:本文主要介绍Windows系统下如何查找JDK的安装路径,文中介绍了三种方法,分别是通过命令行检查、使用verbose选项查找jre目录、以及查看... 目录一、确认是否安装了JDK二、查找路径三、另外一种方式如果很久之前安装了JDK,或者在别人的电脑上,想

java导出pdf文件的详细实现方法

《java导出pdf文件的详细实现方法》:本文主要介绍java导出pdf文件的详细实现方法,包括制作模板、获取中文字体文件、实现后端服务以及前端发起请求并生成下载链接,需要的朋友可以参考下... 目录使用注意点包含内容1、制作pdf模板2、获取pdf导出中文需要的文件3、实现4、前端发起请求并生成下载链接使

Java springBoot初步使用websocket的代码示例

《JavaspringBoot初步使用websocket的代码示例》:本文主要介绍JavaspringBoot初步使用websocket的相关资料,WebSocket是一种实现实时双向通信的协... 目录一、什么是websocket二、依赖坐标地址1.springBoot父级依赖2.springBoot依赖