WebGL學習之紋理盒

原文地址:WebGL學習之紋理盒
我們之前已經學習過二維紋理 gl.TEXTURE_2D,而且還使用它實現(xiàn)了各種效果。但還有一種立方體紋理 gl.TEXTURE_CUBE_MAP刑然,它包含了6個紋理代表立方體的6個面鄙煤。不像常規(guī)的紋理坐標有2個緯度班缰,立方體紋理使用法向量,換句話說三維方向。本節(jié)實現(xiàn)的demo請看 天空盒

3D box

根據(jù)法向量的朝向選取立方體6個面中的一個,這個面的像素用來采樣生成顏色避凝。這六個面通過他們相對于立方體中心的方向被引用。它們是分別是

gl.TEXTURE_CUBE_MAP_POSITIVE_X//右
gl.TEXTURE_CUBE_MAP_NEGATIVE_X//左
gl.TEXTURE_CUBE_MAP_POSITIVE_Y//上
gl.TEXTURE_CUBE_MAP_NEGATIVE_Y//下
gl.TEXTURE_CUBE_MAP_POSITIVE_Z//后
gl.TEXTURE_CUBE_MAP_NEGATIVE_Z//前

環(huán)境貼圖

其實我們更應該把cube map叫作紋理盒眨补,通常紋理盒不是給立方體設置紋理用的管削,設置立方體紋理的標準用法其實是使用二維貼圖,那么紋理盒用來做什么的呢撑螺?紋理盒最常見的用法是用來做環(huán)境貼圖含思。在百度和google地圖中的3D街景就是環(huán)境貼圖應用的一個例子。

紋理

下面是6張紅色峽谷圖片

canyon
canyon

將以上尺寸為512x512的圖片填充到立方體的每個面,以下就是紋理的創(chuàng)建加載過程

// 創(chuàng)建紋理含潘。
var texture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_CUBE_MAP, texture);
 
const faceInfos = [
  {
    target: gl.TEXTURE_CUBE_MAP_POSITIVE_X, 
    url: '/img/sorbin_rt.jpg',
  },
  {
    target: gl.TEXTURE_CUBE_MAP_NEGATIVE_X, 
    url: '/img/sorbin_lf.jpg',
  },
  {
    target: gl.TEXTURE_CUBE_MAP_POSITIVE_Y, 
    url: '/img/sorbin_up.jpg',
  },
  {
    target: gl.TEXTURE_CUBE_MAP_NEGATIVE_Y, 
    url: '/img/sorbin_dn.jpg',
  },
  {
    target: gl.TEXTURE_CUBE_MAP_POSITIVE_Z, 
    url: '/img/sorbin_bk.jpg',
  },
  {
    target: gl.TEXTURE_CUBE_MAP_NEGATIVE_Z, 
    url: '/img/sorbin_ft.jpg',
  },
];
faceInfos.forEach((faceInfo) => {
  const {target, url} = faceInfo;
  // 上傳畫布到立方體貼圖的每個面
  const level = 0;
  const format = gl.RGBA;
  const width = 512;
  const height = 512;
  const type = gl.UNSIGNED_BYTE;
  // 設置每個面饲做,使其立即可渲染
  gl.texImage2D(target, level, format, width, height, 0, format, type, null);
 
  // 異步加載圖片
  const image = new Image();
  image.src = url;
  image.onload = function() {
    // 圖片加載完成將其拷貝到紋理
    gl.bindTexture(gl.TEXTURE_CUBE_MAP, texture);
    gl.texImage2D(target, level, internalFormat, format, type, image);
    gl.generateMipmap(gl.TEXTURE_CUBE_MAP);
  };
});
gl.generateMipmap(gl.TEXTURE_CUBE_MAP);
gl.texParameteri(gl.TEXTURE_CUBE_MAP, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_LINEAR);

法向量

標準立方體法向量 和 紋理盒法向量的區(qū)別


立方體向量

3D立方體使用紋理盒有一個巨大的好處就是不需要額外指定紋理坐標。只要盒子是被放置在世界坐標系的原點调鬓,盒子本身的坐標就可以作為紋理坐標使用艇炎,因為在3D世界中位置本身就是一個向量,表示一個方向腾窝,我們要的就是這個方向缀踪。

所以頂點著色器非常簡單

attribute vec4 a_position;
uniform mat4 u_vpMatrix;
varying vec3 v_normal;

void main() {
    gl_Position = u_vpMatrix * a_position;
    //因為位置是以幾何中心為原點的,可以用頂點坐標作為法向量
    v_normal = normalize(a_position.xyz);
} 

片段著色器中我們需要用samplerCube 代替 sampler2DtextureCube代替texture2DtextureCube 需要vec3類型的向量虹脯。 法向量從頂點著色器傳遞過來經過了插值處理驴娃,需要重新單位化。

precision mediump float; // 從頂點著色器傳入循集。
varying vec3 v_normal; // 紋理唇敞。
uniform samplerCube u_texture; 

void main() {   
  gl_FragColor = textureCube(u_texture, normalize(v_normal));
}

實現(xiàn)

運行后得到如下的效果,很明顯就能看出是個立方體咒彤,并不是我們想要的360度環(huán)繞的3D場景疆柔。


紋理盒

其實我們只需要將相機位置置于原點(0,0,0),同時lookAt向其中的一個面就可以了镶柱。但是在原點有個問題旷档,如果要旋轉查看場景怎么辦?我們可以通過旋轉相機的位置歇拆,這其實就相當于立方體旋轉鞋屈,同時我們不需要矩陣位移相關的信息,只需要方向相關的信息就好了故觅。同時還可以禁止寫入深度緩存厂庇,造成背景在很遠的假象,讓效果更加真實输吏。

const viewPosition = new Vector3([0,0,1]);//相機位置
const lookAt = [0, 0, 0];//原點

//相機繞y軸旋轉
cameraMatrix.rotate(0.2,0,1,0);
viewPoint = cameraMatrix.multiplyVector3(viewPosition);
vpMatrix.setPerspective( 30, canvas.width / canvas.height, 0.1, 5 );
vpMatrix.lookAt(...viewPoint.elements, ...lookAt, 0, 1, 0);

//重置位移
vpMatrix.elements[12] = 0;
vpMatrix.elements[13] = 0;
vpMatrix.elements[14] = 0;

// 禁止寫入深度緩存权旷,造成背景在很遠的假象
gl.depthMask(false);

環(huán)境紋理映射

環(huán)境貼圖還有個更通俗的叫法-天空盒。接著我們還要實現(xiàn)一個非常帥氣的效果贯溅,在天空盒三維場景中拄氯,讓其中的物體反射場景周圍的著色。這個操作就叫做環(huán)境紋理映射(environment mapping)盗迟。

反射

如果物體的表面像光滑的鏡子,那么我們就能看到物體反射出天空和周圍的景色熙含。反射的原理非常簡單罚缕,那就是使用反射公式映射紋理盒對應的紋素:

紋理映射

相機位置(觀察點)和 物體頂點的位置,頂點位置又包含著法線信息怎静,通過GLSL的reflect函數(shù)就可以非常容易的計算反射向量R邮弹,進而確定看到的是哪一塊表面的著色黔衡。

實現(xiàn)

我們就在天空盒下面增加一個鏡面立方體,那就需要增加一對著色器腌乡,首先頂點著色器需要增加法線盟劫,mvp矩陣

attribute vec4 a_position;
attribute vec4 a_normal;
uniform mat4 u_vpMatrix;
uniform mat4 u_modelMatrix;
varying vec3 v_position;
varying vec3 v_normal;

void main() {
    v_position = (u_modelMatrix * a_position).xyz;
    v_normal = vec3(u_modelMatrix * a_normal);
    gl_Position = u_vpMatrix * u_modelMatrix * a_position;
}

片元著色器則需要添加相機位置,紋理以及頂點著色器傳遞過來的法線和頂點位置

precision highp float;
varying vec3 v_position;
varying vec3 v_normal;
uniform samplerCube u_texture;
uniform vec3 u_viewPosition;

void main() {
    vec3 normal = normalize(v_normal);
    vec3 eyeToSurfaceDir = normalize(v_position - u_viewPosition);
    vec3 direction = reflect(eyeToSurfaceDir,normal);
    gl_FragColor = textureCube(u_texture, direction);
}

這樣我們繪制的時候就要輪流切換著色器program

function draw(){
    gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

    //天空盒
    gl.useProgram(program.program);
        //繪制天空盒
    //...

    //立方體
    gl.useProgram(cProgram.program);
        //繪制立方體
    //...

    requestAnimationFrame(draw);
}

最后實現(xiàn)如下效果与纽,demo情況 天空盒

sky box

后記

其實紋理盒除了可以做環(huán)境貼圖侣签,還可以結合光照,陰影貼圖作出很多酷炫的效果急迂。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末影所,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子僚碎,更是在濱河造成了極大的恐慌猴娩,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件勺阐,死亡現(xiàn)場離奇詭異卷中,居然都是意外死亡,警方通過查閱死者的電腦和手機渊抽,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門蟆豫,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人腰吟,你說我怎么就攤上這事无埃。” “怎么了毛雇?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵嫉称,是天一觀的道長。 經常有香客問我灵疮,道長织阅,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任震捣,我火速辦了婚禮荔棉,結果婚禮上,老公的妹妹穿的比我還像新娘蒿赢。我一直安慰自己润樱,他們只是感情好,可當我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布羡棵。 她就那樣靜靜地躺著壹若,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上店展,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天养篓,我揣著相機與錄音,去河邊找鬼赂蕴。 笑死柳弄,一個胖子當著我的面吹牛,可吹牛的內容都是我干的概说。 我是一名探鬼主播碧注,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼席怪!你這毒婦竟也來了应闯?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤挂捻,失蹤者是張志新(化名)和其女友劉穎碉纺,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體刻撒,經...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡骨田,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了声怔。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片态贤。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖醋火,靈堂內的尸體忽然破棺而出悠汽,到底是詐尸還是另有隱情,我是刑警寧澤芥驳,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布柿冲,位于F島的核電站,受9級特大地震影響兆旬,放射性物質發(fā)生泄漏假抄。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一丽猬、第九天 我趴在偏房一處隱蔽的房頂上張望宿饱。 院中可真熱鬧,春花似錦脚祟、人聲如沸谬以。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽为黎。三九已至胡陪,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間碍舍,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工邑雅, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留片橡,地道東北人。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓淮野,卻偏偏與公主長得像捧书,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子骤星,可洞房花燭夜當晚...
    茶點故事閱讀 42,722評論 2 345