Note: This series blog was translated from Nathan Vaughn's Shaders Language Tutorial and has been authorized by the author. If reprinted or reposted, please be sure to mark the original link and description in the key position of the article after obtaining the author’s consent as well as the translator's. If the article is helpful to you, click this Donation link to buy the author a cup of coffee.

说明:该系列博文翻译自Nathan Vaughn着色器语言教程。文章已经获得作者翻译授权,如有转载请务必在取得作者译者同意之后在文章的重点位置标明原文链接以及说明。如果你觉得文章对你有帮助,点击此打赏链接请作者喝一杯咖啡。

朋友们,你们好!在上一篇文章中,我们学习了如何绘制一个圆形并且让它动起来。在这篇文章中,我们将会学习如何绘制一个方块并且用一个旋转矩阵让它转起来。

绘制矩形

绘制一个矩形和绘制一个圆形是非常相似的,我们只需要用另外一个等式。实际上,只需要我们写出一个正确的等式,我们就能几乎能绘制出它对应的图形。矩形的等式如下:

  max(abs(x),abs(y)) = r

  x = x-coordinate on graph
y = y-coordinate on graph
r = radius of square

我们调整等式和变量的顺序,使得等式为0:

  max(abs(x), abs(y)) - r = 0

我们把它改造成下面的样子,就可以Desmos calcultor生成可视化的图形公式了:

max(abs(x), abs(y)) - 2 = 0

复制上面的代码,然后把它粘贴到Desmos calculator中,你就能看到一个半径为2的正方形,它的中心点在原点(0,0)上。

你也可以给它设置一些偏移值:

max(abs(x - offsetX), abs(y - offsetY)) - r = 0

offsetX = how much to move the center of the square in the x-axis
offsetY = how much to move the center of the square in the y-axis

使用像素着色器绘制一个正方形的步骤和我们在上一节教程中讲到的绘制一个圆的步骤十分相似,我们给正方形构建一个指定的函数:

  vec3 sdfSquare(vec2 uv, float size, vec2 offset) {
float x = uv.x - offset.x;
float y = uv.y - offset.y;
float d = max(abs(x), abs(y)) - size; return d > 0. ? vec3(1.) : vec3(1., 0., 0.);
} void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
vec2 uv = fragCoord/iResolution.xy; // <0, 1>
uv -= 0.5; // <-0.5,0.5>
uv.x *= iResolution.x/iResolution.y; // fix aspect ratio vec2 offset = vec2(0.0, 0.0); vec3 col = sdfSquare(uv, 0.2, offset); // Output to screen
fragColor = vec4(col,1.0);
}

哇!我们得到了一个红色正方形 !

让正方形转起来

使用下面的表示法构建一个旋转矩阵,就可以让我们的图形转起来:

矩阵帮助我们处理线性方程线性变换。实际上,一个旋转矩阵就是仿射变换矩阵的一种类型。我们可以使用矩阵来处理其他的放射变换例如:剪切、变形或者反射。

提示: 如果你想要试试矩阵计算,你可以使用Demos Matrix Calculator 或者 WolframAlpha。如果你要复习一下矩阵知识,你可以观看Derek Banas 在YouTuBe上发布的这个神奇的视频

你可以使用我在Desmos上创建的一个可视化旋转图形。我用旋转矩阵创建了一系列的线性方程式,在这些方程式中都使用旋转矩阵。

用旋转矩阵乘以WolfraAlpha中计算后得出的向量[x,y]就能得到线性方程式。其结果就是结果就是一个转换了X和Y坐标的等式。

在Shadertoy中,我们只需要关系旋转矩阵,而不是线性方程式。我也只会在讨论在Desmos中展示旋转示例的时候提及到它。

我们创建一个旋转矩阵函数,这个矩阵接受UV坐标以及一个角度用来旋转正方形。它的返回结果是一个乘以UV坐标后的旋转矩阵。然后,我们在sdfSquare函数中调用rotate函数,给它传递我们的XY坐标,或者一个它的偏移值(如果它存在的话)。我们将使用到iTime作为角度,这样正方形就能动起来。

  vec2 rotate(vec2 uv, float th) {
return mat2(cos(th), sin(th), -sin(th), cos(th)) * uv;
} vec3 sdfSquare(vec2 uv, float size, vec2 offset) {
float x = uv.x - offset.x;
float y = uv.y - offset.y;
vec2 rotated = rotate(vec2(x,y), iTime);
float d = max(abs(rotated.x), abs(rotated.y)) - size; return d > 0. ? vec3(1.) : vec3(1., 0., 0.);
} void mainImage( out vec4 fragColor, in vec2 fragCoord )
{
vec2 uv = fragCoord/iResolution.xy; // <0, 1>
uv -= 0.5; // <-0.5,0.5>
uv.x *= iResolution.x/iResolution.y; // fix aspect ratio vec2 offset = vec2(0.0, 0.0); vec3 col = sdfSquare(uv, 0.2, offset); // Output to screen
fragColor = vec4(col,1.0);
}

请注意我们是如何在Shadertoy中定义矩阵的。让我们更近地观察旋转函数吧。

  vec2 rotate(vec2 uv, float th) {
return mat2(cos(th), sin(th), -sin(th), cos(th)) * uv;
}

根据维基百科上的GLSL规范,我们定义一个由逗号分隔各个值的矩阵,我们需要先看列。因为这个矩阵是mat2类型的,它就是一个2*2的矩阵。前面的两个值表示为第一列,后面两个值表示第二列。在WolframAlpha这种工具中,你需要首先插入行值让后用方括号分隔它们。在使用度量的时候一定要记住这点。

我们的rotate函数返回一个vec2类型的向量,因为2乘2的矩阵(mat2)乘以一个vec2的向量返回的是另外一个vec2向量。

运行以上代码,我们就可以看到一个沿着顺时钟旋转的正方形了。

总结

在这节课中,我们学会了如何绘制一个圆并且使用变形矩阵让它动起来。运用你本节和上一节课学习到的知识,使用一个SDF公式绘制出你自己的2D图形吧!

在下一篇文章中,我将会讨论如何在画布上绘制多种形状,同时也会讨论如何改变背景颜色。

资源

Shadertoy 教程 Part 3 - 矩形和旋转的更多相关文章

  1. HDU 5251 矩形面积 (旋转卡壳)

    2015年百度之星程序设计大赛 - 初赛(1) 1006 比赛链接:2015年百度之星程序设计大赛 - 初赛(1) 题目链接:HDU 5251 Problem Description 小度熊有一个桌面 ...

  2. Shadertoy 教程 Part 2 - 圆和动画

    Note: This series blog was translated from Nathan Vaughn's Shaders Language Tutorial and has been au ...

  3. NeHe OpenGL教程 第四课:旋转

    转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...

  4. BC 2015在百度之星程序设计大赛 - 预赛(1)(矩形区域-旋转卡)

    矩形区域 Accepts: 717 Submissions: 1619 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 ...

  5. BZOJ 1185: [HNOI2007]最小矩形覆盖 [旋转卡壳]

    1185: [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 1435  Solve ...

  6. [hdu5251]矩形面积 旋转卡壳求最小矩形覆盖

    旋转卡壳求最小矩形覆盖的模板题. 因为最小矩形必定与凸包的一条边平行,则枚举凸包的边,通过旋转卡壳的思想去找到其他3个点,构成矩形,求出最小面积即可. #include<cstdio> # ...

  7. bzoj 1185 [HNOI2007]最小矩形覆盖——旋转卡壳

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1185 矩形一定贴着凸包的一条边.不过只是感觉这样. 枚举一条边,对面的点就是正常的旋转卡壳. ...

  8. BZOJ 1185: [HNOI2007]最小矩形覆盖-旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标-备忘板子

    来源:旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标 BZOJ又崩了,直接贴一下人家的代码. 代码: #include"stdio.h" #include"str ...

  9. bzoj1185 [HNOI2007]最小矩形覆盖 旋转卡壳求凸包

    [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec  Memory Limit: 162 MBSec  Special JudgeSubmit: 2081  Solved: 920 ...

随机推荐

  1. [第七篇]——Docker Hello World之Spring Cloud直播商城 b2b2c电子商务技术总结

    Docker Hello World Docker 允许你在容器内运行应用程序, 使用  docker run 命令来在容器内运行一个应用程序. 输出Hello world xxx@xxx:~$ do ...

  2. 在Jupyter Notebook添加代码自动补全功能

    在使用Jupyter notebook时发现没有代码补全功能,于是在网上查找了一些资料,最后总结了以下内容. 1 安装显示目录功能: pip install jupyter_contrib_nbext ...

  3. Java Web实现登录验证码(Servlet+jsp)

    1.生成验证码图片(Servlet) import java.awt.Color; import java.awt.Font; import java.awt.Graphics2D; import j ...

  4. ECSHOP任意页面显示指定分类、数量、排序的任意类型文章,包括只显示置顶or普通的文章

    1.在需要使用此功能的PHP页面里最后的?>前面添加以下代码,现在以article.php为例子 /** jinmozhe 专业ECSHOP二次开发 * 获得指定分类ID.文章类型.指定数量.排 ...

  5. Ubuntu18.04 安装Tomcat 8.5

    下载tomcat,登陆官网:https://tomcat.apache.org/ 点击tar.gz后,弹出这个不大懂这是什么? tomcat要求的jdk版本 解压tar包 sudo tar zxvf ...

  6. css定位,class属性之间有空格与无空格的区别

    中间有空格的情况 是选择到.class1类下的.class2类子节点,即.class2类的节点要是.class1类子节点 <style> .class1 { color: black; } ...

  7. python实现rtsp取流并截图

    import cv2 def get_img_from_camera_net(folder_path): cap = cv2.VideoCapture("rtsp://admin:admin ...

  8. c++ 的学习 第3集-默认参数

    1.默认参数的意思就是  予以形参默认值 2. C++允许函数设置默认参数,在调用时可以根据情况省略实参.规则如下: 默认参数只能按照右到左的顺序 如果函数同时有声明.实现,默认参数只能放在函数声 ...

  9. 鸿蒙内核源码分析(时钟任务篇) | 触发调度谁的贡献最大 | 百篇博客分析OpenHarmony源码 | v3.05

    百篇博客系列篇.本篇为: v03.xx 鸿蒙内核源码分析(时钟任务篇) | 触发调度谁的贡献最大 | 51.c.h .o 任务管理相关篇为: v03.xx 鸿蒙内核源码分析(时钟任务篇) | 触发调度 ...

  10. 前端规范之CSS规范(Stylelint)

    代码规范是软件开发领域经久不衰的话题,几乎所有工程师在开发过程中都会遇到或思考过这一问题.而随着前端应用的大型化和复杂化,越来越多的前端团队也开始重视代码规范.同样,前段时间,笔者所在的团队也开展了一 ...