精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

一起學 WebGL:繪制立方體

開發 前端
現在我們來繪制一個立方體,其實本質和繪制二維圖形是一樣,也是繪制三角形,只是繪制很多個,然后組合起來,作為立方體的幾個面,拼在一起就是一個立方體了。

大家好,我是前端西瓜哥。

之前我們繪制三角形,是一個二維的圖形。

現在我們來繪制一個立方體,其實本質和繪制二維圖形是一樣,也是繪制三角形,只是繪制很多個,然后組合起來,作為立方體的幾個面,拼在一起就是一個立方體了。

繪制三角形,我們用的 API 是:

gl.drawArrays(gl.TRIANGLES, 0, n);

那畫一個立方體,假設選擇 gl.TRIANGLE_FAN 圖元模式,也就是畫 6 個面,每個面有 4 個點。所以我們需要定義 24 個點。

這其實重復定義了頂點,因為一個立方體也就 8 個點,數據是有冗余的。

WebGL 提供了 gl.drawElements() 方法,通過索引值映射的方式來解決這個問題。

首先定義立方體的 8 個頂點(我們命名為 v0 到 v7)的位置和顏色。

const verticesColors = new Float32Array([
  1, 1, 1,     1, 1, 1, // 點 0 白
  -1, 1, 1,    1, 0, 1, // 點 1 品紅
  -1, -1, 1,   1, 0, 0, // 點 2 紅
  1, -1, 1,    1, 1, 0, // 點 3 黃

  1, -1, -1,   0, 1, 0, // 點 4 綠色
  1, 1, -1,    0, 1, 1, // 點 5 青色
  -1, 1, -1,   0, 0, 1, // 點 6 藍色
  -1, -1, -1,  0, 0, 0, // 點 7 黑色
]);

圖片

然后用索引值構造好 6 個面,每個面 2 個三角形:

const indices = new Uint8Array([
  0, 1, 2, 0, 2, 3, // 正面
  0, 3, 4, 0, 4, 5, // 右面
  0, 5, 6, 0, 6, 1, // 上面
  1, 6, 7, 1, 7, 2, // 左面
  7, 4, 3, 7, 3, 2, // 下面
  4, 7, 6, 4, 6, 5, // 背面
]);

和頂點數據類似,索引值也要創建一個緩沖區,并進行綁定,綁定目標變成了 gl.ELEMENT_ARRAY_BUFFER 上。

const indexBuffer = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, indexBuffer);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, indices, gl.STATIC_DRAW);

最后調用 gl.drawElements 方法進行繪制。這里要傳入 indices 數組的長度,WebGL 就會讀取索引值得到對應的頂點信息去一個個繪制三角形啦。

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_BYTE, 0);

繪制結果:

圖片

完整代碼:

// Create a cube
//    v6----- v5
//   /|      /|
//  v1------v0|
//  | |     | |
//  | |v7---|-|v4
//  |/      |/
//  v2------v3

/** @type {HTMLCanvasElement} */
const canvas = document.querySelector('canvas');
const gl = canvas.getContext('webgl');
const infoDiv = document.createElement('div');
document.body.appendChild(infoDiv);

const vertexShaderSrc = `
attribute vec4 a_Position;
attribute vec4 a_Color;
uniform mat4 u_ViewMatrix;  // 視圖矩陣
uniform mat4 u_ProjMatrix; // 正射投影矩陣
mat4 u_MvpMatrix = u_ProjMatrix * u_ViewMatrix;
varying vec4 v_Color;
void main() {
 gl_Position = u_MvpMatrix * a_Position;
 v_Color = a_Color;
}
`;

const fragmentShaderSrc = `
precision highp float;
varying vec4 v_Color;
void main() {
  gl_FragColor = v_Color;
}
`;

/**** 渲染器生成處理 ****/
// 創建頂點渲染器
const vertexShader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vertexShader, vertexShaderSrc);
gl.compileShader(vertexShader);
// 創建片元渲染器
const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(fragmentShader, fragmentShaderSrc);
gl.compileShader(fragmentShader);
// 程序對象
const program = gl.createProgram();
gl.attachShader(program, vertexShader);
gl.attachShader(program, fragmentShader);
gl.linkProgram(program);
gl.useProgram(program);
gl.program = program;

// prettier-ignore
const verticesColors = new Float32Array([
  1, 1, 1,     1, 1, 1, // 點 0 白
  -1, 1, 1,    1, 0, 1, // 點 1 品紅
  -1, -1, 1,   1, 0, 0, // 點 2 紅
  1, -1, 1,    1, 1, 0, // 點 3 黃

  1, -1, -1,   0, 1, 0, // 點 4 綠色
  1, 1, -1,    0, 1, 1, // 點 5 青色
  -1, 1, -1,   0, 0, 1, // 點 6 藍色
  -1, -1, -1,  0, 0, 0, // 點 7 黑色
]);

// prettier-ignore
const indices = new Uint8Array([
  0, 1, 2, 0, 2, 3, // 正面
  0, 3, 4, 0, 4, 5, // 右面
  0, 5, 6, 0, 6, 1, // 上面
  1, 6, 7, 1, 7, 2, // 左面
  7, 4, 3, 7, 3, 2, // 下面
  4, 7, 6, 4, 6, 5, // 背面
]);

// 每個數組元素的字節數
const SIZE = verticesColors.BYTES_PER_ELEMENT;
// 創建緩存對象
const vertexColorBuffer = gl.createBuffer();
const indexBuffer = gl.createBuffer();

gl.bindBuffer(gl.ARRAY_BUFFER, vertexColorBuffer);
gl.bufferData(gl.ARRAY_BUFFER, verticesColors, gl.STATIC_DRAW);

gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, indexBuffer);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, indices, gl.STATIC_DRAW);

// 獲取 a_Position 變量地址
const a_Position = gl.getAttribLocation(gl.program, 'a_Position');
const a_Color = gl.getAttribLocation(gl.program, 'a_Color');
/****** 正射投影 ******/
const u_ViewMatrix = gl.getUniformLocation(gl.program, 'u_ViewMatrix');

// prettier-ignore
const viewMatrix = createViewMatrix(
  3, 4, 8, // 觀察點
  0, 0, 0, // 視點
  0, 1, 0 // 上方向
)
gl.uniformMatrix4fv(u_ViewMatrix, false, viewMatrix);

/****** 正射投影 ******/
const u_ProjMatrix = gl.getUniformLocation(gl.program, 'u_ProjMatrix');
// prettier-ignore
const projMatrix = createPerspective(
  30, canvas.width / canvas.height, 1, 100
)
gl.uniformMatrix4fv(u_ProjMatrix, false, projMatrix);

gl.vertexAttribPointer(a_Position, 3, gl.FLOAT, false, SIZE * 6, 0);
gl.enableVertexAttribArray(a_Position);

gl.vertexAttribPointer(a_Color, 3, gl.FLOAT, false, SIZE * 6, SIZE * 3);
gl.enableVertexAttribArray(a_Color);

/*** 繪制 ***/
// 清空畫布,并指定顏色
gl.clearColor(0, 0, 0, 1);
gl.enable(gl.DEPTH_TEST); // 啟動深度檢測,處理錯誤的像素覆蓋問題

gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
// 繪制三角形
// gl.drawArrays(gl.TRIANGLES, 0, 8);
gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_BYTE, 0);

/************ 后面都是一些工具類方法 ******/

/**** 構造視圖矩陣 ****/
function createViewMatrix(eyeX, eyeY, eyeZ, atX, atY, atZ, upX, upY, upZ) {
  const normalize = (v) => {
    const length = Math.sqrt(v[0] * v[0] + v[1] * v[1] + v[2] * v[2]);
    return [v[0] / length, v[1] / length, v[2] / length];
  };
  const subtract = (v1, v2) => {
    return [v1[0] - v2[0], v1[1] - v2[1], v1[2] - v2[2]];
  };
  const cross = (v1, v2) => {
    return [
      v1[1] * v2[2] - v1[2] * v2[1],
      v1[2] * v2[0] - v1[0] * v2[2],
      v1[0] * v2[1] - v1[1] * v2[0],
    ];
  };

  const zAxis = normalize(subtract([eyeX, eyeY, eyeZ], [atX, atY, atZ]));
  const xAxis = normalize(cross([upX, upY, upZ], zAxis));
  const yAxis = normalize(cross(zAxis, xAxis));

  return new Float32Array([
    xAxis[0],
    yAxis[0],
    zAxis[0],
    0,
    xAxis[1],
    yAxis[1],
    zAxis[1],
    0,
    xAxis[2],
    yAxis[2],
    zAxis[2],
    0,
    -(xAxis[0] * eyeX + xAxis[1] * eyeY + xAxis[2] * eyeZ),
    -(yAxis[0] * eyeX + yAxis[1] * eyeY + yAxis[2] * eyeZ),
    -(zAxis[0] * eyeX + zAxis[1] * eyeY + zAxis[2] * eyeZ),
    1,
  ]);
}

function angleToRadian(angle) {
  return (Math.PI * angle) / 180;
}

/***** 構建透視矩陣 *****/
function createPerspective(fov, aspect, near, far) {
  fov = angleToRadian(fov); // 角度轉弧度
  const f = 1.0 / Math.tan(fov / 2);
  const nf = 1 / (near - far);
  // prettier-ignore
  return new Float32Array([
    f / aspect, 0, 0, 0,
    0, f, 0, 0,
    0, 0, (far + near) * nf, -1,
    0, 0, 2 * far * near * nf, 0,
  ]);
}

線上體驗 demo:

https://codesandbox.io/s/upx8yz?file=/index.js。

責任編輯:姜華 來源: 前端西瓜哥
相關推薦

2023-04-12 07:46:24

JavaScriptWebGL

2023-05-16 07:44:03

紋理映射WebGL

2023-04-11 07:48:32

WebGLCanvas

2023-04-17 09:01:01

WebGL繪制三角形

2023-05-04 08:48:42

WebGL復合矩陣

2023-04-26 07:42:16

WebGL圖元的類型

2023-06-26 15:14:19

WebGL紋理對象學習

2023-03-29 07:31:09

WebGL坐標系

2023-04-13 07:45:15

WebGL片元著色器

2023-05-17 08:28:55

2023-04-27 08:27:29

WebGL變形矩陣

2023-03-02 07:44:39

pixijsWebGL

2023-02-22 09:27:31

CanvasWebGL

2023-05-08 07:29:48

WebGL視圖矩陣

2022-11-29 16:35:02

Tetris鴻蒙

2022-12-02 14:20:09

Tetris鴻蒙

2023-03-30 09:32:27

2022-11-14 17:01:34

游戲開發畫布功能

2019-07-16 16:05:51

PythonScribusRGB

2023-05-06 07:23:57

點贊
收藏

51CTO技術棧公眾號

亚洲欧美在线视频观看| 性欧美xxxx大乳国产app| 91精品一区二区三区在线观看| 国产高清免费在线| 亚洲男人天堂久久| 天堂成人免费av电影一区| 日韩在线视频线视频免费网站| 天堂av.com| 免费毛片b在线观看| 国产精品美女久久久久高潮| 国产精品福利视频| 国产精品无码一区| 欧美视频一区| 怡红院精品视频| 在线播放第一页| 韩国精品主播一区二区在线观看| 亚洲人精品午夜| 精品久久一区二区三区蜜桃| 亚洲天堂视频在线| 在线视频精品| 久久在线免费观看视频| 性欧美13一14内谢| 99国产精品免费网站| 色老综合老女人久久久| 日本免费a视频| 一本一道波多野毛片中文在线 | 欧美激情xxxx性bbbb| 一级黄色性视频| 一区二区三区亚洲变态调教大结局| 色哟哟一区二区| av久久久久久| 日韩成人影视| 亚洲国产精品t66y| 精品毛片久久久久久| 99久久国产热无码精品免费| 视频一区国产视频| 91精品国产91久久久久福利| 中文字幕在线2021| 日韩啪啪电影网| 亚洲美女精品久久| 视频免费在线观看| 免费一区二区三区在线视频| 欧美日本在线视频| 日本久久久久久久久久久久| 亚洲一区站长工具| 精品福利在线观看| 很污的网站在线观看| 在线欧美三级| 日韩一区在线播放| 在线视频亚洲自拍| 婷婷在线视频观看| 国产精品午夜电影| 国产精品久久久对白| 中文字幕第2页| 麻豆精品久久久| 国产精品福利小视频| 少妇太紧太爽又黄又硬又爽| 国产日韩欧美三区| 91黑丝高跟在线| 在线观看国产亚洲| 国产精品尤物| 日韩av毛片网| 中文字幕精品视频在线观看| 久久久久一区| 国产精品福利小视频| 国产九色91回来了| 免费看欧美女人艹b| 国产精品免费网站| 亚洲一二区视频| 久久99国内精品| 亚洲xxxx3d| 亚洲av无码国产综合专区| 国产成人丝袜美腿| 国产综合动作在线观看| 四虎在线免费看| 久久久亚洲精品一区二区三区| 久久青青草原| 国产69精品久久app免费版| 欧美激情综合网| 欧美三级午夜理伦三级老人| 欧美精品videossex少妇| 亚洲高清久久久| 1024av视频| 秋霞国产精品| 69堂成人精品免费视频| 农村末发育av片一区二区 | 国产精品久久久久毛片大屁完整版 | 一区二区三区美女| 国产视频九色蝌蚪| 香蕉视频亚洲一级| 欧美精品黑人性xxxx| 在线观看你懂的视频| 噜噜噜天天躁狠狠躁夜夜精品| 亚洲欧美精品一区二区| 2014亚洲天堂| 亚洲精品美女| 国产精品自拍小视频| 精品人妻伦一区二区三区久久 | 日韩欧美一区二区三区久久| 日韩av手机版| 国产精品调教视频| 正在播放欧美一区| 国产精品999久久久| 日本成人在线不卡视频| http;//www.99re视频| 免费一级在线观看| 亚洲精品久久久蜜桃| 男女曰b免费视频| 欧美日韩视频免费看| 亚洲国产精品女人久久久| 国产福利在线导航| 夜久久久久久| 亚洲一区二区久久久久久久| 男女污污视频在线观看| 亚洲欧美日韩国产一区二区三区| 国产欧美在线一区| 日韩成人在线看| 伊人久久精品视频| 日本一区二区三区免费视频| 久久精品99久久久| 欧美高清视频一区| www.综合网.com| 欧美电影在线免费观看| 中国黄色a级片| 狠狠爱成人网| 成人免费观看网址| 成人资源www网在线最新版| 一区二区三区日韩欧美精品| 超碰超碰在线观看| 一区二区美女| 2019中文字幕全在线观看| 99热这里只有精品99| 欧美激情在线看| 日本久久久精品视频| 风间由美性色一区二区三区四区 | 欧美日韩免费不卡视频一区二区三区| 这里只有精品在线观看视频| 欧美在线高清| 成人观看高清在线观看免费| 成人免费视频| 91久久精品网| 精品少妇人妻一区二区黑料社区| 亚洲激情一区| 国产欧美一区二区在线播放| 欧美xxxx黑人又粗又长| 日韩一二在线观看| 三级影片在线看| 国内精品在线播放| 视频一区二区视频| 亚洲ww精品| 久久精品国产一区二区三区| 国产尤物视频在线观看| 中文字幕中文字幕中文字幕亚洲无线| 亚洲成人av免费看| 日韩欧美视频| 成人黄色在线播放| 黄色在线观看网站| 欧美一级高清大全免费观看| www.5588.com毛片| 国产精品911| 青青在线视频免费观看| 中文字幕一区二区三区日韩精品| 欧美精品在线免费观看| 精品人妻少妇嫩草av无码专区| 亚洲男人电影天堂| 人妻精油按摩bd高清中文字幕| 欧美精品aa| 好吊色欧美一区二区三区| 成人免费网站观看| 亚洲精品网站在线播放gif| 国产精品熟女视频| ㊣最新国产の精品bt伙计久久| 欧美日韩久久婷婷| 午夜日韩视频| 久久亚洲高清| 亚洲电影有码| 久久精品视频亚洲| 成人无码一区二区三区| 欧美色videos| 99精品中文字幕| 国产精品一二一区| 男人靠女人免费视频网站 | 韩国v欧美v日本v亚洲v| 欧美黄网在线观看| 欧美美女在线直播| 国产精品日日摸夜夜添夜夜av| 精品国产99久久久久久| 亚洲精品一区二区在线观看| 九九热在线免费观看| 亚洲国产成人自拍| 不卡的一区二区| 午夜亚洲激情| 日本一本草久p| 欧美人与动xxxxz0oz| 国产精品久久99久久| 宅男网站在线免费观看| 亚洲人成网站在线播| 国产欧美日韩综合精品一区二区三区 | 欧美日本免费一区二区三区| 久久精品国产亚洲AV无码麻豆| 久久色.com| 99精品视频免费版的特色功能| 亚洲激情网址| 自拍另类欧美| 亚洲综合图色| 亚洲一区二区久久久久久| 亚洲欧洲自拍| 久久99精品久久久久久琪琪 | av网站一区二区三区| 日韩一区二区三区久久| 国产亚洲精品bv在线观看| 中文字幕中文字幕一区三区| 亚洲制服欧美另类| 成人动漫视频在线观看完整版| 99久久亚洲国产日韩美女 | 亚洲av鲁丝一区二区三区| 久久九九99视频| 国产大尺度视频| 免费黄网站欧美| 国产精品国产亚洲精品看不卡| 欧美成人激情| 欧美一区二区视频17c| 国产精品久av福利在线观看| 国产中文字幕91| 粉嫩一区二区三区| 97精品伊人久久久大香线蕉| 91麻豆国产福利在线观看宅福利| 国产亚洲欧洲在线| 亚洲欧美日本在线观看| 亚洲成人黄色网址| 国产成人a人亚洲精品无码| 在线看国产日韩| 性无码专区无码| 亚洲国产精品麻豆| 欧美黑人猛猛猛| 国产精品久久久久国产精品日日| 玖草视频在线观看| 9久草视频在线视频精品| 人妻精油按摩bd高清中文字幕| 久久99精品久久久久久动态图| 免费男同深夜夜行网站 | 欧美激情a在线| 成人在线app| www.亚洲一区| 免费在线观看黄| 综合激情国产一区| 成人精品一区二区三区免费| 国产一区二区久久精品| 精品999视频| 亚洲欧洲在线播放| 每日更新在线观看av| 亚洲欧美福利视频| 国产免费av在线| 一区二区国产精品视频| 91精品专区| 中文字幕日韩欧美在线| 日本在线天堂| 久久久成人的性感天堂| 亚洲小说区图片区都市| 欧美国产精品va在线观看| 狂野欧美性猛交xxxxx视频| 欧美激情免费视频| 人成在线免费网站| 国产精品高潮呻吟久久av野狼| 成人全视频免费观看在线看| 国产主播精品在线| 136导航精品福利| 国产精品对白一区二区三区| 欧美一区自拍| 日韩一区二区三区资源| 99精品一区| 日本a在线天堂| 亚洲美女少妇无套啪啪呻吟| 日韩av资源在线| 免费高清成人在线| 伊人五月天婷婷| 成人av高清在线| 日韩人妻一区二区三区| 国产精品网站在线播放| 放荡的美妇在线播放| 午夜av区久久| 久久久久久久亚洲| 91精品一区二区三区久久久久久| 亚洲男人第一天堂| 国产亚洲精品高潮| 97caopor国产在线视频| 91黑丝高跟在线| 九七电影院97理论片久久tvb| 99re视频在线| 欧美猛男男男激情videos| 中文字幕一区二区三区四区五区人| 国产精品www.| 日本在线视频www| 狠狠色狠狠色合久久伊人| 中文视频在线观看| 国产精品久久看| 国产污视频在线看| 欧美三级电影在线观看| 色婷婷在线视频| 中文字幕亚洲一区| av电影免费在线看| 成人日韩在线电影| 久久93精品国产91久久综合| 亚洲成人动漫在线| 日本亚洲视频在线| 亚洲色偷偷色噜噜狠狠99网 | 日韩欧美电影在线| 国模吧精品人体gogo| 久久久久久香蕉网| 视频欧美精品| 欧美日韩精品免费观看视一区二区| 综合日韩在线| 亚洲 欧美 日韩系列| av中文字幕亚洲| 印度午夜性春猛xxx交| 色激情天天射综合网| 囯产精品久久久久久| 日韩亚洲成人av在线| 亚洲精品永久免费视频| 成人免费观看网站| 国产精品99在线观看| 久热免费在线观看| 波多野结衣中文一区| 国产中文av在线| 日本韩国视频一区二区| 国产小视频一区| 免费91在线视频| 国产福利一区二区三区在线播放| 久久狠狠久久综合桃花| 韩国久久久久| 九九九久久久久久久| 国产精品成人网| 国产一卡二卡三卡| 亚洲精品国产精品国产自| 污视频免费在线观看| 国产女同一区二区| 久久国产电影| 激情婷婷综合网| 久久久久九九视频| 四虎成人永久免费视频| 亚洲国模精品私拍| bl视频在线免费观看| 福利精品视频| 国语自产精品视频在线看8查询8| 在线观看视频你懂得| 亚洲美女在线一区| 国产精品区在线观看| www.久久撸.com| 欧美激情三区| 热这里只有精品| 激情六月婷婷综合| 性色av无码久久一区二区三区| 欧美高清精品3d| 麻豆传媒免费在线观看| 成人激情视频在线观看| 成人亚洲一区二区| av网站在线不卡| 国产色一区二区| 国模私拍一区二区| 中文字幕一区二区三区电影| 青青在线精品| 自拍亚洲欧美老师丝袜| 国产乱码精品一区二区三 | 午夜在线视频免费| 18一19gay欧美视频网站| 天堂网av成人| 国语对白做受xxxxx在线中国| 久久久精品影视| 伊人影院中文字幕| 久久久久www| heyzo欧美激情| 37pao成人国产永久免费视频| 久久精品人人做人人爽97 | 欧美一级黄色大片| sm性调教片在线观看| 欧美韩国日本精品一区二区三区| 日韩精品五月天| 欧美爱爱免费视频| 精品国免费一区二区三区| 一区二区三区电影大全| 亚洲视频sss| 国产福利一区在线| 可以免费在线观看的av| 中文字幕日韩av电影| 一区二区三区自拍视频| 黄色片视频在线免费观看| 国产精品毛片a∨一区二区三区| 性生活免费网站| 青草成人免费视频| 五月天综合网站| 精品人妻一区二区三区日产| 欧美在线观看18| 中文在线免费| 欧美视频1区| 国产福利一区二区三区视频 | 欧美日韩黄视频| 成人免费高清观看| 日韩欧美三级电影| 高清久久久久久| 特级西西444www高清大视频|