惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

爱范儿
爱范儿
Know Your Adversary
Know Your Adversary
Google DeepMind News
Google DeepMind News
A
Arctic Wolf
P
Privacy & Cybersecurity Law Blog
云风的 BLOG
云风的 BLOG
Stack Overflow Blog
Stack Overflow Blog
V
Visual Studio Blog
Project Zero
Project Zero
L
LangChain Blog
N
News and Events Feed by Topic
博客园 - Franky
Last Week in AI
Last Week in AI
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
T
The Blog of Author Tim Ferriss
宝玉的分享
宝玉的分享
Scott Helme
Scott Helme
T
The Exploit Database - CXSecurity.com
P
Proofpoint News Feed
Blog — PlanetScale
Blog — PlanetScale
www.infosecurity-magazine.com
www.infosecurity-magazine.com
W
WeLiveSecurity
月光博客
月光博客
博客园_首页
美团技术团队
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
腾讯CDC
Latest news
Latest news
WordPress大学
WordPress大学
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Spread Privacy
Spread Privacy
Attack and Defense Labs
Attack and Defense Labs
量子位
L
LINUX DO - 热门话题
C
CERT Recently Published Vulnerability Notes
Webroot Blog
Webroot Blog
L
Lohrmann on Cybersecurity
aimingoo的专栏
aimingoo的专栏
T
Troy Hunt's Blog
Security Latest
Security Latest
小众软件
小众软件
Cloudbric
Cloudbric
Hacker News: Ask HN
Hacker News: Ask HN
S
Secure Thoughts
雷峰网
雷峰网
T
Threat Research - Cisco Blogs
H
Hacker News: Front Page
IT之家
IT之家
Simon Willison's Weblog
Simon Willison's Weblog

机核

游戏性能旗舰最强之选,一加 Ace 6 至尊版国补到手价2999元起 6元钱自己更换电动车刹车线 《生化危机9:安魂曲》编剧Haris Orkin专访 摸金游戏?音乐游戏!暗区新赛季这把能弹的琴有何来历? 好评国产武侠SRPG《息风谷战略》免费DLC现已推出 | 机核 GCORES 碎片 《生化危机:安魂曲》将于5月实装首个免费更新内容 | 机核 GCORES 新锐东方游戏,谱写世界新章! 沉浸式恋爱视觉小说游戏《心象演算》免费试玩版现已正式上线 | 机核 GCORES 互动影游《代号三国:龙起》上线!穿越三国与曹操并肩、与佳人同行、与权谋博弈! 《老头收集梦想生活》,游戏酒桌会6,录音笔VOL.689 | 机核 GCORES 破界·共生——《白日梦:无限世界》五大核心游戏特点解析 价格已到史低,锐龙5 9600X/锐龙7 9700X正适合抄底 时间循环之旅即刻启程!《归环》一周目测试今日开启 热门在线韩游变魂游,洛奇衍生作能否打破“花瓶”魔咒? LG UltraGear evo 全新高端显示器系列:当“5K”遇见“AI”,不止强大,更懂热爱 反套路三国互动影游《代号三国:龙起》今日上线! 愿望单登记人数突破10万!备受瞩目的“女儿养成游戏” 《梦幻魔法公主》今日于Steam平台上线!限时八折优惠中 《黑神话:悟空》全球音乐会2026巡演将于4月29日12时开票 | 机核 GCORES “Snowguelike”生存肉鸽挖矿新作《蛙穿雪境》公布发售日期,5月7日正式上线 | 机核 GCORES 重塑移动办公、AI创作新境!全新华硕灵耀Air系列、ProArt 骁龙版震撼首发,创芯未来 均分88:《Saros》媒体评分汇总 | 机核 GCORES 上海烛龙公布合作遗迹探险游戏《吉时已到》首支预告片 | 机核 GCORES 首个独立游戏《萝薇日记》已上线Steam! 烛龙新IP《吉时已到》首曝,打造国内首款中式合作遗迹探险游戏 喜加一:《暗黑破坏神Ⅳ》国服现已开启限时免费领取本体活动 | 机核 GCORES 新版《生化危机》电影定于9月18日上映,官方网站现已上线 | 机核 GCORES 《冲就完事模拟器2》“星球大战”联动DLC正式宣布 | 机核 GCORES SteamController将于5月4日发售,售价99美元 | 机核 GCORES 事已至此,内存这么用也算省钱了,“2+1”非对称双通道应用实测 505游戏母公司现已收购《明末:渊虚之羽》IP | 机核 GCORES 基石 手动杂谈12|格斗游戏也能讲好故事 | 机核 GCORES I Love You Mr Snowball 我的向日葵小姐 会比GTA6先发售吗?最硬核的生存游戏《DAYZ》要推新DLC了 周记02:在2026遇到新怪谈 什么硬件,能让游戏Loading界面快速消失? | 机核 GCORES GadioSpec《百年风云世界杯》免费试听集 | 机核 GCORES 百年风云世界杯Vol.1丨足球崛起 | 机核 GCORES 100年前,一群人提前替我们经历了AI恐惧 | 机核 GCORES 《呼啸山庄》2026 ——一辆当代艺术的大卡车冲撞了我的大脑 手游《天穗之咲稻姬:日之香巡灵传》宣布将于7月27日停服 | 机核 GCORES 集结梦之队,征战世界杯,《最佳球会ONLINE》上线Steam 山水绝景随心拼 休闲建造游戏《千里山河录》Steam商店页公开 巫师帽、法袍、魔杖,为什么它们是影视、游戏里的法师必备三件套? 《无鞘信使》-第一章 复古风自动战斗肉鸽《终结之终结》Steam商店页面现已上线 电脑里有一款不破不立的MMO,录音笔VOL.688 | 机核 GCORES 《生化危机》30周年纪念周边发售,这次是真的“保护伞” 可靠耐用+AI全能,惠普战66 2025锐龙版深度体验 《时之书:无尽终章》关卡“大航海时代”全球首次公开 经典名作《乌龙派出所》改编经营模拟游戏《乌龙派出所~阿两的商店街物语~》正式宣布支持简体中文 明日开冲,解锁反套路三国互动影游《代号三国:龙起》即将上线 独立游戏《这是我的宝藏!》已发售~ 降低难度不是唯一解,无压力死亡也是好体验 Netflix官宣新片《普通人》:讲述韩国现代史上的权力风暴 《生化危机:安魂曲》全球销量现已突破700万份 《绝地鸭卫》PC版5月15日正式发售 亡妻回忆录?女性向情感叙事游戏《S-mail》现已正式发售 全新酷黑风格,酷睿Ultra 200S PLUS的高性价比搭档!七彩虹BATTLE-AX B860M-PLUS S WIFI7 V20 超级黑刃主板测评 胖狗 索尼发布了港台地区PS5产品价格调整公告,将于5月1日起实施 二次元怪猎+性感美女!《碧蓝幻想:无尽黄昏》开启Beta公测 这款怪谈类型中式恐怖游戏居然更新了?! 《百日战纪 -最终防卫学园-》改编漫画将于今年冬季开启连载 《如被附身,请致电我们》:匈牙利黑色幽默恐怖小说 《竹屿山房杂部卷五》(译文) 大树 超越引擎 摄影分享丨四月 碎片杂记vol.72 当老式FPS与老式动画技术碰撞出新时代的火花—《神探杰克鼠》 想做独游,如何避免首个项目就褒姒? 在线多人动作游戏《OCTOPinbs》将于5月12日上线Steam! 《酒鬼女神的酒诡》确定将于2026年登陆Steam 《CRYMELIGHT》将于11月5日(周四)正式发售!4月25日开始预购! 钢铁国度MKI部落Evolution,蛮兵部分 【钢铁国度】部落-Primal MK I,蛮兵部分 这是一个高中生用ai跑出来的作品,我自称他为物理神话 钢铁国度MKI部落Metamorphosis,蛮兵部分上 钢铁国度MKI部落Metamorphosis,蛮兵部分下 INDIE Live Expo于4月25日举办:首发9款新作,超200款独立游戏亮相 暗黑卡牌策略新作《魔忌:穷鼠啮狸》发布全新中文试玩版和发售预告片,4月30日正式上线在即 战锤40K长篇小说:变节者・苦难主宰(三)(全书完) 战锤40K长篇小说:变节者・苦难主宰(二) 战锤40K长篇小说:变节者・苦难主宰(一) 英语语言学习丨短语专题1:短语的特点 唯一的EVE战争 【少前同人】【M200】战术人形会梦见音乐会吗 【昏迷3】即将发售,前作主角悉数到场!“恶灵”宋老师化身可操控角色 业内人士:游戏公司“十有八九”使用生成式AI,包括卡普空 欢庆一周年:《光与影:33号远征队》全新纪念艺术图、超值折扣与免费更新同步上线 才刚刚开始呢 【抽奖】《星际卡车司机》推出免费大型更新,四折平史低折扣进行中 四人合作FPS游戏《佣兵猎手》抢先体验重大更新 1 现已上线 Raw Fury新作《深馅地牢 Deep Dish Dungeon》将于今秋加入 XGP 塔防幸存者游戏《魔怪来袭》推出首个 DLC 《饿狼传说:群狼之城》1周年纪念!新DLC“沃尔夫冈·克劳萨”今日参战 猫狗相伴 欢乐闯关 双人合作平台跳跃游戏《猫狗同行》上线Steam商店页 《同行:月球逃脱》(Together: Moon Escape)上线将于明日上线Steam
Shader 魔法的学习之路(3):笑脸威力加强版
mnikn · 2022-05-17 · via 机核

上篇文章中我们画出来了一个笑脸,但是看起来很粗糙,这次我们玩大一点,画一个能够让别人看不出是用代码画出来的笑脸吧!

一张脸简化为脸、眼睛、嘴巴和眉毛,我们就依次去画吧!

本次文章内容可能看上去有点复杂,不懂的同学可以自己模拟一下,调一下参数慢慢思考。

先画脸

首先我们先按照之前的方式设置一下基础的代码:

vec4 face(vec2 uv) { vec4 color = vec4(1.0, 0.6, 0.3, 1.0); float d = length(uv); color.a = smoothstep(0.5, 0.495, d); // 边缘渐变 float gradient = smoothstep(0.45, 0.5, d); // 多次相乘让 gradient 能够让颜色从中心到边缘的衰弱感加深, 参考 y=x^2 函数图形 gradient *= gradient; color.rgb = mix(color.rgb, color.rgb * 0.6, gradient); return color; } ​ void mainImage( out vec4 fragColor, in vec2 fragCoord ) { vec4 color = vec4(0.0, 0.0, 0.0, 1.0); ​ vec2 uv = fragCoord/iResolution.xy; uv -= 0.5; // 让画布变为正方形 uv.x *= iResolution.x / iResolution.y; vec4 face_c = face(uv); color.rgb = mix(color.rgb, face_c.rgb, face_c.a); fragColor = color; }

代码和之前相似,不过不同点是:

  • 我们添加了 uv.x *= iResolution.x / iResolution.y 让我们整个画布能够标准化成正方形,而上次文章因为没有这步,画布是长方形,所以画出来的是一个椭圆。

  • 使用了 mix 函数而不是直接相乘来混色,mix 函数的作用是接受前两个参数,第三个参数作为比例,根据比例返回处于前两个参数之间的值。使用 mix 函数而不是直接相乘的原因,是因为接下来我们要画很多个圆,直接相乘的话会让这些圆之间的颜色相互干扰,而 mix 函数用 alpha 做比例保证之间不会相互影响。

  • 我们多添加了脸部边缘的渐变,其中我们处理完后利用 y=x^2 的函数图形让这个值快速上升,从而让边缘颜色衰退不是线性的,衰退感更强。

接下来我们逐步添加细节:

float sat(float t) { return clamp(t, 0.0, 1.0); } ​ float remap01(float a, float b, float t) { return sat((t - a) / (b - a)); } ​ float remap(float a, float b, float c, float d, float t) { return remap01(a, b, t) * (d - c) + c; } ​ vec4 face(vec2 uv) { vec4 color = vec4(1.0, 0.6, 0.3, 1.0); float d = length(uv); color.a = smoothstep(0.5, 0.495, d); // 边缘渐变 float gradient = smoothstep(0.45, 0.5, d); // 多次相乘让 gradient 能够让颜色从中心到边缘的衰弱感加深, 参考 y=x^2 函数图形 gradient *= gradient; color.rgb = mix(color.rgb, color.rgb * 0.6, gradient); // 脸部高光 float highlight = smoothstep(0.45, 0.447, d); // 使用 remap 让高光只画在脸的上半部分 highlight *= remap(0.45, 0.05, 0.8, 0.0, uv.y); color.rgb = mix(color.rgb, vec3(1.0), highlight); // 描边 color.rgb = mix(color.rgb, vec3(0.5, 0.1, 0.1), smoothstep(0.487, 0.495, d)); // 脸颊 // 让脸颊能够镜像 uv.x = abs(uv.x); vec2 cheek_uv = uv + vec2(-0.2, 0.2); float cheek_d = length(cheek_uv); float cheek = smoothstep(0.2, 0.05, cheek_d) * 0.3; color.rgb = mix(color.rgb, vec3(0.9, 0.2, 0.1), cheek); return color; }

这里我们添加了几个帮助函数:

  • sat:clamp 的进一步封装,返回值限制在 [0,1],超过或者小于这个范围就限制为 1 或者 0

  • remap01:mix 的相反版本,根据前两个参数的相差比例,把第三个参数固定在 [0,1] 滑动

同时我们添加了脸部的上边高光,描边和脸颊,思路和之前类似,一些特殊操作在代码上有相关说明。

画个眼睛

vec2 within(vec2 uv, vec4 rect) { return (uv - rect.xy) / (rect.zw - rect.xy); } ​ vec4 eye(vec2 uv, float dir) { uv -= 0.5; uv.x *= dir; // 基础的圆 vec4 color = vec4(1.0); float d = length(uv); color.a = smoothstep(0.4, 0.39, d); vec3 iris_color = vec3(0.2, 0.6, 0.9); // 眼眶 float iris = smoothstep(0.27, 0.4 , d) * 0.7; iris *= iris; color.rgb = mix(color.rgb, iris_color, iris); // 眼眶下左边缘描边 color.rgb *= 1.0 - smoothstep(0.38, 0.4, d) * sat(-uv.x-uv.y); // 眼球描边 color.rgb = mix(color.rgb, vec3(0.0), smoothstep(0.27, 0.26, d)); // 眼球 iris_color *= 1.0 + smoothstep(0.26, 0.05, d); color.rgb = mix(color.rgb, iris_color, smoothstep(0.26, 0.24, d)); // 瞳孔 color.rgb = mix(color.rgb, vec3(0.0), smoothstep(0.15, 0.14, d)); // 瞳孔高光 float highligh1 = smoothstep(0.12, 0.11, length(uv+vec2(-0.2,-0.1))); color.rgb = mix(color.rgb, vec3(1.0), highligh1); float highlight2 = smoothstep(0.07, 0.06, length(uv+vec2(0.15,0.08))); color.rgb = mix(color.rgb, vec3(1.0), highlight2); return color; } ​ void mainImage( out vec4 fragColor, in vec2 fragCoord ) { // ...之前的代码 // 画眼球 vec4 eye_c = eye(within(vec2(abs(uv.x), uv.y), vec4(0.04, -0.12, 0.35, 0.22)), dir); color.rgb = mix(color.rgb, eye_c.rgb, eye_c.a); }

大体的思路和画脸的时候差不多,有一些特殊的是我们添加了一个 within 函数,该函数的作用是把画布限制在对应区域,然后我们就可以在 eye 函数里面认为 uv 整个范围是 [0,1],不用再仔细调坐标。

注意一点是,在调用 within 时,我们先让 uv 做了水平镜像然后再传进函数,而不是在函数里面做镜像,因为 eye 函数里面画布已经被限制区域了,如果在函数里面做镜像,就只会在限制区域内做镜像而不是基于脸的中心做镜像。

同时我们使用了个 sign 函数来获得 uv.x 当前是负数还是正数,然后传进函数改变眼睛绘制方向。因为两只眼睛是做了水平镜像,这样眼睛高光的方向是向着脸中心,而我们想要眼睛高光统一向着右边,所以乘上 uv.x 的方向来修正。

画嘴

vec4 mouth(vec2 uv) { uv -= 0.5; vec4 color = vec4(0.8, 0.3, 0.2, 1.0); // 嘴 uv.y += uv.x*uv.x; float d = length(uv); color.a = smoothstep(0.4, 0.39, d); // 牙齿 float td = remap(-0.25, 0.25, 0.0, 1.0, uv.y); vec3 teeth_col = vec3(smoothstep(0.5, 0.3, d)); color.rgb = mix(color.rgb, teeth_col, smoothstep(0.05, 0.02, td)); // 嘴部渐变 float mtd = 1.0 - remap(-0.25, 0.3, 0.0, 1.0, uv.y); color.rgb *= 1.0 - smoothstep(0.4, 0.0, mtd) * 0.2; return color; } ​ void mainImage( out vec4 fragColor, in vec2 fragCoord ) { // ...之前的代码 // 画嘴 vec4 mouth_c = mouth(within(uv, vec4(-0.3, -0.12, 0.3, -0.45))); color.rgb = mix(color.rgb, mouth_c.rgb, mouth_c.a); }

画眉毛

vec4 brow(vec2 uv) { uv -= 0.5; // 调整眉毛位置 uv.y -= uv.x * uv.x - uv.x; vec4 color = vec4(0.8, 0.4, 0.2, 1.0) * 0.8; float d = length(uv); color *= 1.0 + smoothstep(0.38, 0.05, d); color.a = smoothstep(0.4, 0.39, d); // 描边 color.rgb *= 1.0 - smoothstep(0.37, 0.4, d) * 0.5; return color; } ​ void mainImage( out vec4 fragColor, in vec2 fragCoord ) { // ...之前的代码 // 画眉毛 vec4 brow_c = brow(within(vec2(abs(uv.x), uv.y), vec4(0.04, 0.33, 0.36, 0.24))); color.rgb = mix(color.rgb, brow_c.rgb, brow_c.a); }

总结

这次文章里面的内容较多,但是所有的画法主要是绘制位置的计算和颜色间的各种混合,大家最好自己调一下代码消化一下。