news 2026/4/18 11:54:27

【URP】Unity[后处理]色调分离SplitToning

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【URP】Unity[后处理]色调分离SplitToning

核心功能与用途

‌视觉风格化‌:将阴影和高光区域分离着色,常见于电影调色(如《银翼杀手2049》的橙青色调)或游戏场景氛围营造

‌色彩对比增强‌:通过互补色强化画面层次感,例如阴影用冷色(蓝)、高光用暖色(橙)

‌性能优势‌:属于低开销的色彩校正类后处理,适合移动端使用

发展历史

起源自传统胶片摄影的化学调色工艺,后引入数字图像处理

Unity早期通过Amplify Color等插件实现,2018年后整合到Post Processing Stack v2中,现为URP/HDRP标准组件

原理

SplitToning是Unity URP后处理中用于实现色调分离效果的技术,其核心原理是通过对图像的高光和阴影区域分别应用不同的颜色映射,从而创造出艺术化的色彩分级效果。以下是详细解析:

底层原理

‌颜色分离机制‌

SplitToning将图像像素按亮度分为高光(亮部)和阴影(暗部)两部分,通过阈值控制分离范围。高光区域应用_SplitToningHighlightsColor,阴影区域应用_SplitToningShadowsColor,中间过渡区域通过平滑插值混合。

‌LUT(颜色查找表)支持‌

URP可能结合LUT技术加速颜色映射。LUT将输入颜色值映射到预定义输出值,SplitToning的色调映射可通过LUT贴图(如1024x32尺寸)高效实现,每个Tile对应不同的亮度区间。

‌Shader实现流程‌

采样原始图像像素并计算亮度(如使用Luminance()函数)

根据亮度值选择高光或阴影颜色

应用平滑过渡(如smoothstep函数)避免硬边界

最终输出混合结果。

示例说明

以下是一个简化的Shader代码片段,展示SplitToning的核心逻辑:

hlsl

float3 ApplySplitToning(float3 inputColor, float3 shadowsColor, float3 highlightsColor, float balance) {

float luminance = Luminance(inputColor);

float t = smoothstep(0.2, 0.8, luminance); // 过渡区间控制

float3 shadows = lerp(inputColor, shadowsColor * inputColor, 1.0 - t);

float3 highlights = lerp(inputColor, highlightsColor * inputColor, t);

return lerp(shadows, highlights, balance); // 平衡参数调节整体倾向

}

‌参数说明‌:

shadowsColor/highlightsColor:阴影/高光的目标色调(如蓝色高光+橙色阴影)

balance:控制整体偏向高光或阴影(0.5为均衡)。

性能优化建议

‌与URP管线集成‌

通过RenderFeature将SplitToning作为后处理阶段插入渲染管线,注意在Volume组件中配置参数以实现动态调整。

‌结合SRP Batcher‌

若自定义Shader,需确保符合SRP Batcher要求(如使用CBUFFER封装变量),以减少DrawCall开销。

‌LUT优化‌

使用256x16的小尺寸LUT贴图可降低带宽占用,但可能损失精度;1024x32适合高质量需求。

典型应用场景包括电影感调色(如《银翼杀手》风格的冷色调高光+暖色调阴影)或风格化游戏渲染

参数详解与用例

参数 含义 典型用例

Shadows 阴影区域色调(RGB) 暗部填充冷色(如#1E3A8A)增强景深

Highlights 高光区域色调(RGB) 亮部使用暖色(如#F59E0B)模拟阳光

Balance 阴影/高光混合权重 正值偏向高光,负值强化阴影(-20~20)

URP实现流程

SplitToningExample.cs

using UnityEngine;

using UnityEngine.Rendering;

using UnityEngine.Rendering.Universal;

public class SplitToningExample : VolumeComponent, IPostProcessComponent {

[Tooltip("阴影色调")] public ColorParameter shadows = new ColorParameter(Color.blue);

[Tooltip("高光色调")] public ColorParameter highlights = new ColorParameter(Color.yellow);

[Tooltip("平衡值")] public ClampedFloatParameter balance = new ClampedFloatParameter(0f, -20f, 20f);

public bool IsActive() => shadows.value != Color.gray || highlights.value != Color.gray;

public bool IsTileCompatible() => false;

}

SplitToningRenderPass.cs

using UnityEngine;

using UnityEngine.Rendering;

using UnityEngine.Rendering.Universal;

public class SplitToningRenderPass : ScriptableRenderPass {

private Material material;

private SplitToningExample settings;

public SplitToningRenderPass(Material mat) {

material = mat;

renderPassEvent = RenderPassEvent.BeforeRenderingPostProcessing;

}

public void Setup(SplitToningExample volSettings) {

settings = volSettings;

if (settings != null)

material.SetColor("_Shadows", settings.shadows.value);

material.SetColor("_Highlights", settings.highlights.value);

material.SetFloat("_Balance", settings.balance.value);

}

public override void Execute(ScriptableRenderContext context, ref RenderingData data) {

CommandBuffer cmd = CommandBufferPool.Get("SplitToning");

Blit(cmd, ref data, material, 0);

context.ExecuteCommandBuffer(cmd);

CommandBufferPool.Release(cmd);

}

}

SplitToningFeature.cs

using UnityEngine;

using UnityEngine.Rendering.Universal;

public class SplitToningFeature : ScriptableRendererFeature {

private SplitToningRenderPass pass;

public Material effectMaterial;

public override void Create() {

pass = new SplitToningRenderPass(effectMaterial);

}

public override void AddRenderPasses(ScriptableRenderer renderer, ref RenderingData data) {

var stack = VolumeManager.instance.stack;

var settings = stack.GetComponent<SplitToningExample>();

if (settings.IsActive()) {

pass.Setup(settings);

renderer.EnqueuePass(pass);

}

}

}

使用步骤

‌创建Volume Profile‌

Hierarchy右键 → Volume → Global Volume

添加SplitToningExample组件

‌Shader实现‌

hlsl

// Shader核心算法

half3 ApplySplitToning(half3 color, half3 shadows, half3 highlights, half balance) {

half luminance = Luminance(color);

half t = saturate(luminance - balance * 0.01);

return lerp(shadows, highlights, t) * color;

}

‌效果调试‌

阴影色调:适用于地下城/夜晚场景(#2E1065)

高光色调:适合沙漠/黄昏(#F97316)

Balance:-10使画面更阴沉,+10增强阳光感

性能优化建议

避免与Bloom等高开销效果叠加使用

移动端建议使用LUT(颜色查找表)替代实时计算

通过Local Volume按需启用(如仅在过场动画使用)

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 7:01:40

再谈需求无止境,EAST和金融机构--SMP(软件制作平台)

金融机构对其监管机关的各类监管要求是非常重视的&#xff0c;涉及处罚的监管文件固然重要&#xff0c;但是监管机关要求金融机构报送的各类监管数据也是一个重要的方面&#xff0c;而且这些数据也直接关系到监管机构对金融机构的综合评价。 EAST通常指的是Examination and An…

作者头像 李华
网站建设 2026/4/18 11:13:47

Fabulously Optimized Minecraft模组包完整使用指南

Fabulously Optimized是一个专注于性能提升和图形增强的Minecraft模组包&#xff0c;通过精心挑选的优化模组组合&#xff0c;为玩家提供流畅且视觉震撼的游戏体验。该模组包整合了多种性能优化工具和视觉增强组件&#xff0c;让低配置设备也能享受高品质的Minecraft游戏。 【免…

作者头像 李华
网站建设 2026/4/17 17:52:42

Gridfinity模块化收纳系统:从零开始打造完美工作台

Gridfinity模块化收纳系统&#xff1a;从零开始打造完美工作台 【免费下载链接】gridfinity-rebuilt-openscad A ground-up rebuild of the stock gridfinity bins in OpenSCAD 项目地址: https://gitcode.com/gh_mirrors/gr/gridfinity-rebuilt-openscad 还在为工作台上…

作者头像 李华
网站建设 2026/4/18 4:57:35

一次由 DNS 反解析引发的 SpringBoot 启动卡顿问题

TL;DR使用 114 DNS 时&#xff0c;反解析内网 IP 无响应&#xff0c;导致 SpringBoot 启动时 Liquibase 初始化阻塞约 30 秒InetAddress.getLocalHost() 获取到的可能是127.0.0.1&#xff0c;而非实际的内网 IP&#xff08;如 192.168.x.x&#xff09;现象SpringBoot启动时&…

作者头像 李华
网站建设 2026/4/18 7:55:34

跟我学C++中级篇——循环展开的分

一、循环展开 什么是循环展开&#xff1f;为什么要循环展开&#xff1f;可能对不少的程序员来说&#xff0c;循环展开,loop unrolling。它是一种编译器优化的技术&#xff0c;它的本质是通过减少条件分支和运行时的pipeline停顿来提升性能&#xff0c;并有可能提高缓存的命中率…

作者头像 李华
网站建设 2026/4/18 8:00:41

OpenChat成本优化终极指南:AI聊天机器人费用节约完整教程

OpenChat成本优化终极指南&#xff1a;AI聊天机器人费用节约完整教程 【免费下载链接】OpenChat OpenChat - 一个用户聊天机器人控制台&#xff0c;简化了大型语言模型的利用&#xff0c;支持创建和管理定制聊天机器人。 项目地址: https://gitcode.com/gh_mirrors/op/OpenCh…

作者头像 李华