当前位置: 首页 > news >正文

Unity Shader示例 6: 卡渲基础 - 描边 + 着色

0 、获取原神模型:

【游戏开发实战】下载原神模型,PMX转FBX,导入到Unity中,卡通渲染,绑定人形动画(附Demo工程)-CSDN博客

《原神》公测视频征集计划
 

一、描边pass:Outline

1. 将顶点沿着法线方向向外扩

2. 剔除正面,只保留背面,得到外轮廓:Cull Front

3. 控制粗细随视距(顶点与摄像机距离)缩放:

等价于之前做的连连看中 “乘上depth” 的操作,视距越小 东西也要变小
v2f vert(appdata v)
 {
    v2f o;
    
    //随视距缩放
    float camDistance = length(mul(unity_ObjectToWorld, v.vertex) -             
    _WorldSpaceCameraPos);//顶点与摄像机的距离
    float3 OutlineWidth = v.normal * _OutlineWidth * camDistance;

    //顶点外扩
    float3 pos = v.vertex + OutlineWidth;

    o.pos = TransformObjectToHClip(pos); 
    o.uv = TRANSFORM_TEX(v.uv, _MainTex);
    return o;
}

未描边
添加描边

选加:

4. 利用模板缓冲,仅获得最外轮廓

5. 利用法线平均化修复法线转折处的描边断裂

6. SRP合批优化:render object过滤绘制外轮廓的pass

二、着色Pass:Unlit

1. 兰伯特 + rampTexture ——二分

2. 高光 + 边缘光(菲涅尔)

3. 阴影:① 产生阴影的Pass:ShadowCaster,② 再在Unlit中接收阴影
               可以在urp管线的配置中修改shadow的精度

 half4 frag(v2f i) : SV_Target
{
    half4 col = (0,0,0,0);
    half3 l = _MainLightPosition;
    half3 v = normalize(i.viewWS);

    //漫反射
    half nl = dot(l, i.normalWS) * 0.5 + 0.5 ;

    //阴影
    if(_UseShadow){
        Light mainLight = GetMainLight(i.shadowCoord);
        nl *= mainLight.shadowAttenuation * 0.5 + 0.5;//获取阴影衰减值
    }
    half4 baseColor = tex2D(_MainTex, i.uv);
    col = baseColor;

    //RampTex
    if(_UseRampTex){
        half4 rampColor = tex2D(_RampTex, nl);
        col *= rampColor;
    }
               
    //镜面反射
    if(_UseSpecular){
        half3 h = normalize(l + v);
        half nh = pow(max(0, dot(i.normalWS, h)), _SpecPower * 100);
        col += nh * _SpecColor;
    }
               
    //边缘光(菲涅尔)
    if(_UseRimLight){
        half nv = 1 - max (0, dot (i.normalWS, v));
        half f = pow(nv, _RimPower) * _SpecColor;
        col += f;
    }

    return col ; 
}

http://www.dtcms.com/a/21967.html

相关文章:

  • 地基Spring中bean生命周期和设计模式
  • 介绍cherrypick
  • 【第1章:深度学习概览——1.6 深度学习框架简介与选择建议】
  • 面试题总结
  • 安装vite报错Install for [ ‘create-vite@latest‘ ] failed with code 1
  • 本地 Ollama 部署 Deepseek R1 并使用 Spring AI Alibaba 构建 Chat 应用示例
  • K8s学习总结
  • Langchain对管道操作符|的重构实现链式流程
  • nodejs 038: Js Worker线程消息传递 Worker.postMessage() Worker.postMessage()
  • Django快速入门
  • 【论文投稿】Python 网络爬虫:探秘网页数据抓取的奇妙世界
  • 归并排序的应用—计算逆序对的个数
  • 使用 pgvector 实现 PostgreSQL 语义搜索和 RAG:完整指南
  • 长视频生成、尝试性检索、任务推理 | Big Model Weekly 第56期
  • 为AI聊天工具添加一个知识系统 之107 详细设计之48 理解和角色
  • 从零到上线:Node.js 项目的完整部署流程(包含 Docker 和 CICD)
  • Spring Boot 项目启动报错 “找不到或无法加载主类” 解决笔记
  • 【ISO 14229-1:2023 UDS诊断全量测试用例清单系列:第五节】
  • MySQL数据库三:操作数据库(二)
  • 【ISO 14229-1:2023 UDS诊断全量测试用例清单系列:第十节】
  • mac docker镜像加速正确配置方式
  • 【MySQL常见疑难杂症】常见文件及其所存储的信息
  • 尚硅谷爬虫note005
  • 基于Knative的无服务器引擎重构:实现毫秒级冷启动的云原生应用浪潮
  • 数据结构笔记之时间复杂度O(n)中的O是什么的缩写,为什么要用O这个字母?
  • 快速设置 Docker 网络代理配置
  • 手机ROM是什么
  • 网络安全|网络安全学习方法
  • 【办公类-90-02】】20250215大班周计划四类活动的写法(分散运动、户外游戏、个别化综合)(基础列表采用读取WORD表格单元格数据,非采用切片组合)
  • 自己动手实现一个简单的Linux AI Agent