three.js常用材质:基本材质兰伯特材质冯氏材质标准材质

我们可以自己使用着色器实现这些材质,用于批量渲染等用途。

为了简单,假设物体只有一张漫反射贴图,场景中只存在一个环境光和一个平行光。

一、基本材质(MeshBasicMaterial)

基本材质不对光源产生反应。

顶点着色器

varying vec2 vUv;

void main() {
vUv = uv; vec3 transformed = vec3( position );
vec4 mvPosition = modelViewMatrix * vec4( transformed, 1.0 );
gl_Position = projectionMatrix * mvPosition;
}

片源着色器

uniform vec3 diffuse;
uniform float opacity; uniform sampler2D map; varying vec2 vUv; void main() {
vec4 diffuseColor = vec4( diffuse, opacity ); vec4 texelColor = texture2D( map, vUv );
diffuseColor *= texelColor; gl_FragColor = diffuseColor;
}

二、兰伯特材质(MeshLambertMaterial)

兰伯特材质只有漫反射,没有高光。

顶点着色器

uniform vec3 directColor; // 平行光颜色
uniform vec3 directDirection; // 平行光方向 #define PI 3.14159265359 varying vec2 vUv;
varying vec3 vLightFront; void main() {
vUv = uv; vec3 transformedNormal = normalMatrix * vec3( normal ); vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
gl_Position = projectionMatrix * mvPosition; float dotNL = dot( normalize( transformedNormal ), directDirection );
vLightFront = clamp( dotNL, 0.0, 1.0 ) * PI * directColor;
}

片源着色器

uniform vec3 diffuse; // 物体颜色
uniform float opacity; // 透明度 uniform sampler2D map; uniform vec3 ambientColor; // 漫反射光颜色 varying vec2 vUv;
varying vec3 vLightFront; // 双向反射PI
#define RECIPROCAL_PI 0.31830988618 void main() {
vec4 diffuseColor = vec4( diffuse, opacity ); vec4 texelColor = texture2D( map, vUv );
diffuseColor *= texelColor; // 出射光 = 直接漫反射 + 间接漫反射
vec3 outgoingLight = vLightFront + ambientColor * RECIPROCAL_PI * diffuseColor.rgb; gl_FragColor = vec4( outgoingLight, diffuseColor.a );
}

三、冯氏材质(MeshPhongMaterial)

冯氏材质很重要的两个属性是高光颜色(specular)光亮度(shininess)

顶点着色器

varying vec2 vUv;
varying vec3 vNormal;
varying vec3 vViewPosition; void main() {
vUv = uv;
vNormal = normalize( normalMatrix * normal ); vec3 transformed = vec3( position );
vec4 mvPosition = modelViewMatrix * vec4( transformed, 1.0 );
vViewPosition = - mvPosition.xyz; gl_Position = projectionMatrix * mvPosition;
}

片源着色器

// 参考资料:
// BRDF-双向反射分布函数:https://baike.baidu.com/item/双向反射分布函数/22311036
// 常见的三个光照模型:Lambert,Phong,BlinnPhong:https://blog.csdn.net/taoqilin/article/details/52800702
// 菲涅尔公式:https://baike.baidu.com/item/菲涅耳公式/9103788
// 菲涅尔折射率:https://baike.baidu.com/item/菲涅尔折射率/2712906 uniform vec3 diffuse; // 物体颜色
uniform float opacity; // 透明度
uniform vec3 specular; // 高光颜色
uniform float shininess; // 光亮度 uniform sampler2D map; uniform vec3 ambientColor; // 漫反射光颜色
uniform vec3 directColor; // 平行光颜色
uniform vec3 directDirection; // 平行光方向 varying vec2 vUv;
varying vec3 vNormal;
varying vec3 vViewPosition; // 双向反射PI
#define RECIPROCAL_PI 0.31830988618 // 菲涅尔反射
vec3 F_Schlick( const in vec3 specularColor, const in float dotLH ) {
float fresnel = exp2( ( -5.55473 * dotLH - 6.98316 ) * dotLH );
return ( 1.0 - specularColor ) * fresnel + specularColor;
} // Blinn-Phong光照模型
float D_BlinnPhong( const in float shininess, const in float dotNH ) {
return RECIPROCAL_PI * ( shininess * 0.5 + 1.0 ) * pow( dotNH, shininess );
} void main() {
// 物体颜色
vec4 diffuseColor = vec4( diffuse, opacity ); vec4 texelColor = texture2D( map, vUv );
diffuseColor *= texelColor; // 环境光漫反射(BRDF兰伯特漫反射)
vec3 indirectDiffuse = ambientColor * RECIPROCAL_PI * diffuseColor.rgb; // 法线
vec3 normal = normalize( vNormal ); // 平行光漫反射(BRDF兰伯特漫反射)
float dotNL = clamp( dot( normal, directDirection ), 0.0, 1.0 );
vec3 irradiance = dotNL * directColor;
vec3 directDiffuse = irradiance * RECIPROCAL_PI * diffuseColor.rgb; // 平行光镜面反射
vec3 halfDir = normalize( directDirection + normalize( vViewPosition ) ); // 半角向量
float dotNH = clamp( dot( normal, halfDir ), 0.0, 1.0 );
float dotLH = clamp( dot( directDirection, halfDir ), 0.0, 1.0 );
vec3 F = F_Schlick( specular, dotLH ); // 菲涅尔反射
float D = D_BlinnPhong( shininess, dotNH ); // Blinn-Phong光照模型
vec3 directSpecular = F * ( 0.25 * D ); // 出射光 = 环境光漫反射 + 平行光漫反射 + 平行光镜面反射
vec3 outgoingLight = indirectDiffuse + directDiffuse + directSpecular; gl_FragColor = vec4( outgoingLight, diffuseColor.a );
}

四、标准材质(MeshStandardMaterial)

标准材质也叫物理材质或pbr材质,很重要的两个属性是金属度(metalness)粗糙度(roughness)

顶点着色器

varying vec2 vUv;
varying vec3 vNormal;
varying vec3 vViewPosition; void main() {
vUv = uv;
vNormal = normalize( normalMatrix * vec3( normal ) ); vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
vViewPosition = - mvPosition.xyz; gl_Position = projectionMatrix * mvPosition;
}

片源着色器

// 参考资料:
// BRDF-双向反射分布函数:https://baike.baidu.com/item/双向反射分布函数/22311036
// 基于物理的渲染—更精确的微表面分布函数GGX: https://www.jianshu.com/p/be4f025aeb3c
// 菲涅尔公式:https://baike.baidu.com/item/菲涅耳公式/9103788
// 菲涅尔折射率:https://baike.baidu.com/item/菲涅尔折射率/2712906
// Moving Frostbite to Physically Based Rendering 3.0: https://blog.csdn.net/wodownload2/article/details/103126247 uniform vec3 diffuse; // 物体颜色
uniform float opacity; // 透明度
uniform float metalness; // 金属度
uniform float roughness; // 粗糙度 uniform sampler2D map; uniform vec3 ambientColor; // 漫反射光颜色
uniform vec3 directColor; // 平行光颜色
uniform vec3 directDirection; // 平行光方向 varying vec2 vUv;
varying vec3 vNormal;
varying vec3 vViewPosition; // 双向反射PI
#define RECIPROCAL_PI 0.31830988618 // 菲涅尔反射
vec3 F_Schlick( const in vec3 specularColor, const in float dotLH ) {
float fresnel = exp2( ( -5.55473 * dotLH - 6.98316 ) * dotLH );
return ( 1.0 - specularColor ) * fresnel + specularColor;
} float G_GGX_SmithCorrelated( const in float alpha, const in float dotNL, const in float dotNV ) {
float a2 = pow2( alpha );
float gv = dotNL * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNV ) );
float gl = dotNV * sqrt( a2 + ( 1.0 - a2 ) * pow2( dotNL ) );
return 0.5 / max( gv + gl, EPSILON );
} // 微表面分布函数
float D_GGX( const in float alpha, const in float dotNH ) {
float a2 = pow2( alpha );
float denom = pow2( dotNH ) * ( a2 - 1.0 ) + 1.0;
return RECIPROCAL_PI * a2 / pow2( denom );
} vec3 BRDF_Specular_GGX( const in vec3 directDirection, const in vec3 normal, const in viewDir, const in vec3 specularColor, const in float roughness ) {
float alpha = pow2( roughness );
vec3 halfDir = normalize( directDirection + viewDir );
float dotNL = clamp( dot( normal, directDirection ), 0.0, 1.0 );
float dotNV = clamp( dot( normal, viewDir ), 0.0, 1.0 );
float dotNH = clamp( dot( normal, halfDir ), 0.0, 1.0 );
float dotLH = clamp( dot( directDirection, halfDir ), 0.0, 1.0 );
vec3 F = F_Schlick( specularColor, dotLH );
float G = G_GGX_SmithCorrelated( alpha, dotNL, dotNV );
float D = D_GGX( alpha, dotNH );
return F * ( G * D );
} void main() {
vec4 diffuseColor = vec4( diffuse, opacity ); vec4 texelColor = texture2D( map, vUv );
diffuseColor *= texelColor; // 法线
vec3 normal = normalize( vNormal ); // 环境光
vec3 indirectDiffuse = ambientColor * RECIPROCAL_PI * diffuseColor.rgb * ( 1.0 - metalness ); // 间接漫反射 // 平行光
float dotNL = clamp( dot( normal, directDirection ), 0.0, 1.0 );
vec3 irradiance = dotNL * directColor;
vec3 specularColor = mix( vec3( 0.04 ), diffuseColor.rgb, metalness ); vec3 directDiffuse = irradiance * RECIPROCAL_PI * diffuseColor.rgb * ( 1.0 - metalness ); // 直接漫反射
vec3 directSpecular = irradiance * BRDF_Specular_GGX( directDirection, normal, normalize( vViewPosition ), specularColor, clamp( roughness, 0.04, 1.0 ) ); // 直接镜面反射 // 出射光 = 间接漫反射光 + 直接漫反射 + 直接镜面反射光
vec3 outgoingLight = indirectDiffuse + directDiffuse + directSpecular; gl_FragColor = vec4( outgoingLight, diffuseColor.a );
}

四种材质完整实现源码:https://gitee.com/tengge1/ShadowEditor/tree/master/ShadowEditor.Web/src/render/shader/material_simple

参考资料

1. 基于three.js的开源三维场景编辑器https://github.com/tengge1/ShadowEditor
7. Moving Frostbite to Physically Based Rendering 3.0: https://blog.csdn.net/wodownload2/article/details/103126247
 

three.js各种材质的实现源码的更多相关文章

  1. JS魔法堂:jsDeferred源码剖析

    一.前言 最近在研究Promises/A+规范及实现,而Promise/A+规范的制定则很大程度地参考了由日本geek cho45发起的jsDeferred项目(<JavaScript框架设计& ...

  2. 原生JS研究:学习jquery源码,收集整理常用JS函数

    原生JS研究:学习jquery源码,收集整理常用JS函数: 1. JS获取原生class(getElementsByClass) 转自:http://blog.csdn.net/kongjiea/ar ...

  3. 如何优雅的阅读 GitHub 上开源 js 框架和库的源码

    如何优雅的阅读 GitHub 上开源 js 框架和库的源码 step 先总后分,即先了解一下啊框架的大体架构,又一个全局的认识,在选择某些和感兴趣的部分,仔细阅读,各个击破: 带着问题阅读,用到了什么 ...

  4. jquery自定义插件结合baiduTemplate.js实现异步刷新(附源码)

    上一篇记录了BaiduTemplate模板引擎使用示例附源码,在此基础上对使用方法进行了封装 自定义插件jajaxrefresh.js 代码如下: //闭包限定命名空间 (function ($) { ...

  5. JS魔法堂:剖析源码理解Promises/A规范

    一.前言 Promises/A是由CommonJS组织制定的异步模式编程规范,有不少库已根据该规范及后来经改进的Promises/A+规范提供了实现 如Q, Bluebird, when, rsvp. ...

  6. JS魔法堂:mmDeferred源码剖析

    一.前言 avalon.js的影响力愈发强劲,而作为子模块之一的mmDeferred必然成为异步调用模式学习之旅的又一站呢!本文将记录我对mmDeferred的认识,若有纰漏请各位指正,谢谢.项目请见 ...

  7. 百度地图 api 功能封装类 (ZMap.js) 本地搜索,范围查找实例 [源码下载]

    相关说明 1. 界面查看: 吐槽贴:百度地图 api 封装 的实用功能 [源码下载] 2. 功能说明: 百度地图整合功能分享修正版[ZMap.js] 实例源码! ZMap.js 本类方法功能大多使用 ...

  8. 百度地图 api 功能封装类 (ZMap.js) 新增管理事件功能 [源码下载]

    ZMap 功能说明 ZMap.js 本类方法功能大多使用 prototype 原型 实现: 包含的功能有:轨迹回放,圈画区域可编辑,判断几个坐标是否在一个圆圈内,生活服务查询,从经纬度获取地址信息,地 ...

  9. 【 js 性能优化】【源码学习】underscore throttle 与 debounce 节流

    在看 underscore.js 源码的时候,接触到了这样两个方法,很有意思: 我先把实现的代码撂在下面,看不懂的可以先跳过,但是跳过可不是永远跳过哦- 一个是 throttle: _.throttl ...

随机推荐

  1. python习题——随机整数生成类

    随机整数生成类 可以先设定一批生成数字的个数,可设定指定生成的数值的范围 1.普通类实现 import random import random class RandomGen: def __init ...

  2. canvas与工作流的不解之缘

    html的标签 <canvas>用于图形的绘制,通过脚本 (通常是JavaScript)来完成,canvas简而言之就是个画布.上一篇文章我们提到工作流的一个重要组成部分:流程建模,也就是 ...

  3. python面向对象<三>

    类属性.实例属性: class Tool(object): #属性(类属性)类对象(Tool) num = 0 #方法 def __init__(self,new_name): self.name = ...

  4. python函数的基本语法<二>

    函数的流程控制: if...else... a = 100 b = 200 if a == 100 and b ==300: print('100,200') elif b == 200: print ...

  5. MySQL每个分类的前几条记录

    MySQL 获取所有分类和每个分类的前几条记录 比如有文章表 Article(Id,Category,InsertDate) 现在要用SQL找出每种类型中时间最新的前N个数据组成的集合 SELECT ...

  6. 利用Xshell配置ssh免密码登录虚拟机,进行虚拟机与物理机的传输

    先说一下 Xshell如何无密连接虚拟机: ssh登录提供两种认证方式:口令(密码)认证方式和密钥认证方式.其中口令(密码)认证方式是我们最常用的一种,这里介绍密钥认证方式登录到linux的方法.使用 ...

  7. ZeroC ICE中的对象模型和概念

    Ice对象的模型和概念. Ice Object并非是我们的接口实现类的实例对象.我们的接口实现类的实例对象只是充当Ice Object的Servant的角色.一个Ice Object可以有众多Serv ...

  8. web前端面试题总结(html、css)

    1.对 WEB 标准以及 W3C 的理解与认识? 参考: 标签闭合.标签小写.不乱嵌套.提高搜索机器人搜索几率.使用外 链 css 和 js 脚本. 结构行为表现的分离.文件下载与页面速度更快.内容能 ...

  9. mysql安装与sqlyog安装

    首先是mysql安装,参考下面两个链接 https://www.cnblogs.com/gengyufei/p/11735358.html#_label4 https://www.jb51.net/a ...

  10. day01_爬虫和数据

    1.什么是爬虫 1.1.爬虫的定义   脚本,程序--->自动抓取万维网上信息的程序. 1.2.爬虫的分类 ​ 2.1.通用爬虫 ​ 通用网络爬虫 是 捜索引擎抓取系统(Baidu.Google ...