Shadertoy 教程 Part 3 - 矩形和旋转
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图形吧!
在下一篇文章中,我将会讨论如何在画布上绘制多种形状,同时也会讨论如何改变背景颜色。
资源
- 2D Rotation Example
- Vector and Matrix Operations in GLSL
- Vector and Matrix Transformations
- Matrix Arithmetic
- 2D SDFs
Shadertoy 教程 Part 3 - 矩形和旋转的更多相关文章
- HDU 5251 矩形面积 (旋转卡壳)
2015年百度之星程序设计大赛 - 初赛(1) 1006 比赛链接:2015年百度之星程序设计大赛 - 初赛(1) 题目链接:HDU 5251 Problem Description 小度熊有一个桌面 ...
- Shadertoy 教程 Part 2 - 圆和动画
Note: This series blog was translated from Nathan Vaughn's Shaders Language Tutorial and has been au ...
- NeHe OpenGL教程 第四课:旋转
转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...
- BC 2015在百度之星程序设计大赛 - 预赛(1)(矩形区域-旋转卡)
矩形区域 Accepts: 717 Submissions: 1619 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 ...
- BZOJ 1185: [HNOI2007]最小矩形覆盖 [旋转卡壳]
1185: [HNOI2007]最小矩形覆盖 Time Limit: 10 Sec Memory Limit: 162 MBSec Special JudgeSubmit: 1435 Solve ...
- [hdu5251]矩形面积 旋转卡壳求最小矩形覆盖
旋转卡壳求最小矩形覆盖的模板题. 因为最小矩形必定与凸包的一条边平行,则枚举凸包的边,通过旋转卡壳的思想去找到其他3个点,构成矩形,求出最小面积即可. #include<cstdio> # ...
- bzoj 1185 [HNOI2007]最小矩形覆盖——旋转卡壳
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1185 矩形一定贴着凸包的一条边.不过只是感觉这样. 枚举一条边,对面的点就是正常的旋转卡壳. ...
- BZOJ 1185: [HNOI2007]最小矩形覆盖-旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标-备忘板子
来源:旋转卡壳法求点集最小外接矩形(面积)并输出四个顶点坐标 BZOJ又崩了,直接贴一下人家的代码. 代码: #include"stdio.h" #include"str ...
- bzoj1185 [HNOI2007]最小矩形覆盖 旋转卡壳求凸包
[HNOI2007]最小矩形覆盖 Time Limit: 10 Sec Memory Limit: 162 MBSec Special JudgeSubmit: 2081 Solved: 920 ...
随机推荐
- Redis核心原理与实践--列表实现原理之ziplist
列表类型可以存储一组按插入顺序排序的字符串,它非常灵活,支持在两端插入.弹出数据,可以充当栈和队列的角色. > LPUSH fruit apple (integer) 1 > RPUSH ...
- urllib库爬虫技术从0开学习
urllib库 urllib库是pytho中一个最基本网络请求库.可以模拟浏览器的行为,向指定的服务器发送一个请求,并可以保存服务器返回的数据. urllopen函数 在python的urllib库中 ...
- java的split方法中的regex参数
我们需要以|进行分割,为了匹配|本身,正则中采用\|进行转义,而Java中\也表示转义,从java到正则需要必须使用\\|进行转义,即split中的\\表示正则的转义.
- 一文详解JavaScript的继承模式
1 原型链继承 #### ES6中通过原型继承多个引用类型的属性和方法,由于原型和实例的关系,即每个构造函数都有自己的原型对象,同时原型有一个属性指向构造函数,并且实例有一个内部的指针指向原型.如果存 ...
- 433MHZ SPI模块使用心得
最近使用了433MHZ的模块进行了一个通讯项目,选用的是SX1208模块,对接了RTOS和Linux两个操作系统,使用心得如下: 1. 首先要拿来datasheet看一遍,通揽一下它的功能.可以得到一 ...
- 最新seo优化技巧
国内的SEO也发展不少年份了.我是最早开始从事SEO的那一班人.看着这个行业从零开始发展,长大.成熟还谈不上.可以这样说吧,国内做这个行业的,高手并不多.实战的高手更是寥寥无几.当然这个是我个人的推断 ...
- Jmeter扩展组件开发(6) - 将响应结果数据显示到查看结果树中
CODE //用来存储响应数据,目的是将响应结果放到查看结果树当中private String resultData;/** 这个方法就是实现你具体功能逻辑的方法* @param javaSample ...
- jmeter性能实战
概述 性能测试: 通过并发工具请求服务器,提前发现性能问题,优化并解决 为什么做性能测试? 常规需求 用户反馈性能问题 项目对性能不放心 性能测试的最终目标:? 性能指标分析 多快好省 项目性能场景提 ...
- 使用form插件 和ajax 结合使用 没有调用success的原因
当我做文件上传的时候出现不调用success方法 将datype: 'json' 注释掉后成功响应 浏览器显示200状态码,并且响应头为json格式,格式ajax不认为它是json,所以一直执行错误 ...
- 51nod1836-战忽局的手段【期望dp,矩阵乘法】
正题 题目连接:http://www.51nod.com/Challenge/Problem.html#problemId=1836 题目大意 \(n\)个点\(m\)次随机选择一个点标记(可以重复) ...