fabric 材質定義:着色器實踐app
貼到沙盒裏就能夠運行:函數
var viewer = new Cesium.Viewer("cesiumContainer"); viewer.scene.globe.depthTestAgainstTerrain = true; viewer.camera.setView({ destination : new Cesium.Cartesian3(-2644963.9889313546, 5763731.142118295, 2199400.7089496767), //世界座標系下的一個座標點 orientation : {//旋轉角度 heading :6.075, pitch :-0.727, roll : 6.283 } }); const extrudedPolygon = new Cesium.PolygonGeometry({ polygonHierarchy : new Cesium.PolygonHierarchy( Cesium.Cartesian3.fromDegreesArray([ 112.41726298378288, 23.290411251106182, 113.67072522399741, 23.560312361463682, 114.09370956893551, 22.590768298743153, 112.83803246418894, 22.285610818885644 ]) ), extrudedHeight: 3000 }); const instance = new Cesium.GeometryInstance({ geometry: extrudedPolygon, id: 'box with height' }); const m = new Cesium.Material({ fabric: { type: 'Color', uniforms: { color: new Cesium.Color(216 / 255.0, 170 / 255.0, 208 / 255.0).withAlpha(0.618), }, } }); const aper = new Cesium.MaterialAppearance({ material : m, }); var p = viewer.scene.primitives.add(new Cesium.Primitive({ geometryInstances: instance, appearance: aper, releaseGeometryInstances: false, compressVertices: false, })); // p.readyPromise.then(v => console.log(v)); const vs = aper.vertexShaderSource; const fs = aper.fragmentShaderSource; const fs2 = aper.getFragmentShaderSource(); console.log(`// 頂點着色器: ${vs}`); console.log(`// 片元着色器: ${fs}`); console.log(`// 片元着色器2: ${fs2}`);
const m = new Cesium.Material({ fabric: { source: `float a = 12.0;`, } });
隨便定義一個浮點數,發現報錯:prototype
加上 uniform
限定字,報錯稍微改了一點:3d
因此,這個 source 是有規則的。code
我改爲這樣:component
source: `czm_material czm_getMaterial(czm_materialInput materialInput) { }`,
報錯變化了:orm
大意是指,czm_getMaterial
這個函數沒有返回值。這很正常,強類型的 GLSL 規定了這個函數的返回值類型是結構體 czm_material
,那麼再次修改它。對象
source: `czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); return material; }`,
這時,形狀有顏色了:blog
material 這個變量是一個結構體,經過修改其材質因子便可實現材質修改。圖片
修改其漫反射因子:
注意,glsl 中建立結構體 vec3 的默認值是 (0, 0, 0),如今我想要個粉色,rgb色值是:(216 / 255.0, 170 / 255.0, 208 / 255.0),即 (0.8470588235294118, 0.66666666, 0.8156862745098039)
source: `czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); material.diffuse = vec3(0.8470588235294118, 0.66666666, 0.8156862745098039); return material; }`,
沒毛病,顏色出來了:
你能夠在不少個地方獲取材質、外觀的着色器源代碼:
上面在 fabric 對象中的 source 屬性指定的 glsl 源代碼,與 console.log(m.shaderSource) 出來的是徹底同樣的,因此此處忽略。
當經過 2.2 節中對漫反射屬性的設置後,外觀對象的 vertexShaderSource
和 fragmentShaderSource
輸出結果以下:
// 頂點着色器: attribute vec3 position3DHigh; attribute vec3 position3DLow; attribute vec3 normal; attribute vec2 st; attribute float batchId; varying vec3 v_positionEC; varying vec3 v_normalEC; varying vec2 v_st; void main() { vec4 p = czm_computePosition(); v_positionEC = (czm_modelViewRelativeToEye * p).xyz; // position in eye coordinates v_normalEC = czm_normal * normal; // normal in eye coordinates v_st = st; gl_Position = czm_modelViewProjectionRelativeToEye * p; }
// 片元着色器: varying vec3 v_positionEC; varying vec3 v_normalEC; varying vec2 v_st; void main() { vec3 positionToEyeEC = -v_positionEC; vec3 normalEC = normalize(v_normalEC); #ifdef FACE_FORWARD normalEC = faceforward(normalEC, vec3(0.0, 0.0, 1.0), -normalEC); #endif czm_materialInput materialInput; materialInput.normalEC = normalEC; materialInput.positionToEyeEC = positionToEyeEC; materialInput.st = v_st; czm_material material = czm_getMaterial(materialInput); #ifdef FLAT gl_FragColor = vec4(material.diffuse + material.emission, material.alpha); #else gl_FragColor = czm_phong(normalize(positionToEyeEC), material, czm_lightDirectionEC); #endif }
觀察片元着色器代碼中的主函數,其中有一句調用 czm_material material = czm_getMaterial(materialInput);
這一句即是對咱們在 fabric 對象中寫入的 glsl 代碼的調用。
最終,進入 #ifdef FLAT
分支(才疏學淺,不知道這個 FLAT 宏是什麼),對像素着色,使用 material 結構的漫反射因子 + 自發光因子 + 透明度因子進行疊加,生成最終的顏色值。因此,這個時候不妨回到 Material 的 source 中,繼續動手腳。
是存在直接修改 Appearance 對象 fragmentShader、vertexShader 的大佬的,後面有機會展開說說。
source: `czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); material.diffuse = vec3(0.8470588235294118, 0.66666666, 0.8156862745098039); material.specular = 1.0; material.shininess = 0.8; return material; }`,
(我偷偷把高度設爲了 30000,否則不太明顯)
換個地圖和參數:
稍微有那麼一點感受了。
const m = new Cesium.Material({ translucent: false, fabric: { source: `czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); material.diffuse = vec3(0.24313725490196078, 0.7372549019607844, 0.9333333333333333); material.specular = 0.5; material.shininess = 0.8; material.emission = vec3(0.0, 0.66666666, 0.0); return material; }`, } });
用的是 1. 中的代碼,修改 aper 對象的構造參數,直接將 2.3 中的片元着色器代碼貼入看看:
const aper = new Cesium.MaterialAppearance({ fragmentShaderSource: ` varying vec3 v_positionEC; varying vec3 v_normalEC; varying vec2 v_st; void main() { vec3 positionToEyeEC = -v_positionEC; vec3 normalEC = normalize(v_normalEC); #ifdef FACE_FORWARD normalEC = faceforward(normalEC, vec3(0.0, 0.0, 1.0), -normalEC); #endif czm_materialInput materialInput; materialInput.normalEC = normalEC; materialInput.positionToEyeEC = positionToEyeEC; materialInput.st = v_st; czm_material material = czm_getMaterial(materialInput); #ifdef FLAT gl_FragColor = vec4(material.diffuse + material.emission, material.alpha); #else gl_FragColor = czm_phong(normalize(positionToEyeEC), material, czm_lightDirectionEC); #endif } ` });
沒有問題:
因此,基於此模板,只要膽大心細(劃掉)只要對 Cesium 內置的着色器足夠了解,徹底能夠本身改頂點和片元着色器。
const aper = new Cesium.MaterialAppearance({ fragmentShaderSource: ` varying vec3 v_positionEC; varying vec3 v_normalEC; varying vec2 v_st; void main() { vec3 positionToEyeEC = -v_positionEC; vec3 normalEC = normalize(v_normalEC); #ifdef FACE_FORWARD normalEC = faceforward(normalEC, vec3(0.0, 0.0, 1.0), -normalEC); #endif czm_materialInput materialInput; materialInput.normalEC = normalEC; materialInput.positionToEyeEC = positionToEyeEC; materialInput.st = v_st; czm_material material = czm_getMaterial(materialInput); material.diffuse = vec3(0.24313725490196078, 0.7372549019607844, 0.9333333333333333); material.emission = vec3(0.0, 0.66666666, 0.0); material.specular = 0.5; material.shininess = 0.8; #ifdef FLAT gl_FragColor = vec4(material.diffuse + material.emission, material.alpha); #else gl_FragColor = czm_phong(normalize(positionToEyeEC), material, czm_lightDirectionEC); #endif } ` });
const apr = new Cesium.MaterialAppearance({ vertexShaderSource: ` attribute vec3 position3DHigh; attribute vec3 position3DLow; attribute vec3 normal; attribute vec2 st; attribute float batchId; varying vec3 v_positionEC; varying vec3 v_normalEC; varying vec2 v_st; void main() { float zh = position3DHigh.z * 0.97; float zl = position3DLow.z * 0.97; vec3 th = vec3(position3DHigh.xy, zh); vec3 tl = vec3(position3DLow.xy, zl); vec4 p = czm_translateRelativeToEye(th, tl); v_positionEC = (czm_modelViewRelativeToEye * p).xyz; // position in eye coordinates v_normalEC = czm_normal * normal; // normal in eye coordinates v_st = st; gl_Position = czm_modelViewProjectionRelativeToEye * p; } `, })
依舊是上方 1. 的例子,只不過在頂點着色器稍微動動手腳,可達到變形的效果:
很惋惜這個 position3DHigh 和 position3DLow 並非這個 Primitive 的局部相對座標,因此直接修改 z = 0 是壓不平的,可是從圖中可略見端倪,猜想這個 z 值是世界座標,後續使用半透明地形看看。
uniform
在 WebGL 中就是恆定值的意思。通常 WebGL 用 attribute
關鍵字指定頂點屬性或外來值,用 uniform
關鍵字指定常量,用 varying
關鍵字指定頂點着色器、片元着色器共享的變量。
在 Cesium fabric 規則中,fabric.uniforms 的全部變量,在 fabric.source 中能夠直接使用。
例如,我須要傳入一個透明度:
const m = new Cesium.Material({ fabric: { uniforms: { my_var: 0.5, }, source: ` czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); material.diffuse = vec3(0.5, 0.9, 0.3); material.alpha = my_var; return material; } ` } });
是能夠的:
打印 這個 m變量,能夠輕鬆看到 glsl 代碼:
而且支持直接對 js 的變量進行修改以從新着色:
m.uniforms.my_var = 0.9;
uniforms 是 fabric 對象的一個屬性,按理說,你能夠給這個對象傳遞任何與 glsl 內置結構、數據類型有對應關係的數據,例如上例的 my_var
,是數字類型,在着色器內部自動解析爲 uniform float my_var_0;
參考官方給出的二十多種預置 Material,若是你有興趣,能夠直接把它們的 source 打印出來觀察。
例如,在鏡面反射材質中,它的 uniforms 就有這兩個:
uniforms : { image : 'specular.png', channel : 'a' }
一個是圖片路徑,一個是圖片用於鏡面反射強度的通道(此處是 alpha 通道)。
若是你傳遞的是對象,例如最多見的紋理材質中:
uniforms: { image: 'diffuse.png', my_struct: { x: 10, y: 2 } }
這個 my_struct,最終會傳入一個結構體 uniform vec2 my_struct_0;
固然有的時候不要做死,好比這個狀況是轉譯不了的:
uniforms: { my_var: 0.5, my_struct: { x: 12, y: 5, name: { value: 'aaa' }, obj: false, time: 5 } }
會報錯,由於 my_struct 已經超出了 glsl 能理解的類型。
事實上,你在 uniforms 對象中寫的任何數據,在 fabric.components 中同樣能用,而且 Cesium 的內置結構體常量、函數都是能夠直接使用的。
從着色器的角度看,一種材質無非就是 czm_getMaterial()
函數的返回值罷了。
這裏僅僅改的是材質,屬於片元着色器階段發生的事情,在第四節中已經看到了 Material 中寫的着色器代碼是如何被 Appearance 對象的片元着色器代碼調用的。若是你想修改 Primitive 的形狀,那就要去修改 Appearance 的頂點着色器。
Primitive 是 Scene 對象下的,在渲染循環中它是最終被宰殺的羔羊(劃掉),只要是 Primitive,只要你有能力去修改它的着色器代碼,就能夠自定義不少東西出來。
將紋理座標做爲漫反射顏色寫入,就能看到紋理座標的樣子了:
const m = new Cesium.Material({ fabric: { uniforms: { my_var: 0.5, }, source: ` czm_material czm_getMaterial(czm_materialInput materialInput) { czm_material material = czm_getDefaultMaterial(materialInput); material.diffuse = vec3(materialInput.st, 0.0); material.alpha = my_var; return material; } ` } });